Java之== 和 equals()的区别

本文详细解析了Java中==与equals的区别,==用于比较基本数据类型的值和引用类型的地址,而equals则用于比较对象的内容。文章通过示例代码及运行结果展示了不同情况下的比较方式,并介绍了字符串常量池的概念。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

== 和equals

简单概括就是:

==比较基本数据类型的值和引用类型对象的引用地址,equals比较对象的内容。


下面我们具体来说一下这两个:

==:

我们知道java在内存中有栈(stack)和堆(heap)之分。栈中存放基本数据类型和引用类型的变量,堆中存放对象的实际内容。当我们通过new实例化一个对象时,在内存是这样存储对象的:

Student s = new Student("jack",13);

    s持有对实际对象的引用。所以引用类型的数据通过==比较的其实对象就是在内存的地址,但是对象如果不是父子关系的话不能用==来进行比较。

    基本数据类型都是保存在栈中,所以基本数据类型的==比较的就是他们的值,没有类型的限制:
int i = 65;
float f = 65.0f;
char c = 'A';
System.out.println(i == f);
System.out.println(f == c);

    运行结果:


    可以看到这个三个对象==的结果都是true,因为他们都是基本数据类型,只要值相同,==得到的结果就会返回true。

equals():
equals()方法是Object类中的一个方法,Object提供这个方法的本意是让子类重写equals()方法( 重写equals()方法相关可以参考这篇博客),如果没有进行重写的话,则还是用父类的equals()方法,我们看下Object中的equals()方法:
public boolean equals(Object obj) {
    return (this == obj);
}

    可以看到,Object中的equals()内部还是用的==来判断。


    看一下下面的这段代码,想一下他们的输出结果是什么?
package com.ljw.StringStringBuilderStringBuffer;

/**
 * Created by liujiawei on 2018/7/3.
 */
public class TestEquals {
    public static void main(String[] args) {

        String a = "ab";
        String b = "ab";
        String c = new String("ab");
        String d = new String("ab");

        System.out.print("a==b的结果是: ");
        System.out.println(a == b);
        System.out.print("a==c的结果是: ");
        System.out.println(a == c);
        System.out.print("c==d的结果是: ");
        System.out.println(c == d);
        System.out.print("ab equals的结果是: ");
        System.out.println(a.equals(b));
        System.out.print("ac equals的结果是: ");
        System.out.println(a.equals(c));

    }
}

    运行结果:


结果和你想的一样么?

在解释上面的结果之前,要先介绍一个概念:
常量池(constant pool),在方法区中,用来保存关于类,接口,方法中的常量,包括字符串常量。

知道了常量池以后,我们用内存示意图来说明上面的几个string对象的存储,相信看完以后,对上面的结果就可以理解了:
第一步:
String a = "ab";

首先在栈中存放了变量名a,因为这是字符串常量,所以去常量中找“ab”,没有找到,创建了“ab”。

第二步:
String b = "ab";

    栈中创建变量b,去常量池中找“ab”,发现已经被创建过了,所以b直接指向它。

第三步:
String c = new String("ab");

    栈中创建c,通过String的构造方法实例化的对象,在堆中创建对象,再指向方法区中的“ab”

第四步:
String d = new String("ab");

    栈中创建d,堆中创建实例化的对象,指向方法区的“ab”。


    看完上面的图,几个==的判断相信结果一目了然,至于equals的结果则是因为String类重写了equals()方法,看下具体源码:

public boolean equals(Object anObject) {
    if (this == anObject) {
        return true;
    }
    if (anObject instanceof String) {
        String anotherString = (String) anObject;
        int n = value.length;
        if (n == anotherString.value.length) {
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = 0;
            while (n-- != 0) {
                if (v1[i] != v2[i])
                        return false;
                i++;
            }
            return true;
        }
    }
    return false;
}
    可以看到,在==判断以后,会转去判断对象具体的内容,所以abcd四个用equals()比较得到的结果都会返回true。


<think>首先,用户的问题是:“我想理解Java中'==''equals()'的区别 请问Java中运算符==方法equals()区别”。我需要基于提供的引用来回答这个问题。 回顾引用内容: - 引用[1]:讨论了关系运算符,包括==,但它主要关注基本数据类型的比较,如int a=1,b=2; System.out.println(a==b); //false。这显示==用于基本数据类型的值比较。 - 引用[2]:展示了三元运算符的使用,但提到了name.equals("哪吒"),这涉及到equals()方法用于字符串比较。 - 引用[3]:直接比较了Java String中的==equals()。它总结说:==比较字符串对象的地址,而equals()比较字符串的内容。还提到通过new创建的字符串,内容相同但地址不同,因此应使用equals()。 - 引用[4]:讨论了equal函数(可能是equals()==区别。它说equals()比较对象的值(包括属性),而==比较堆中的内存地址。还提到在重写equals()时,需要检查是否为同一对象。 用户的问题是关于==equals()区别,所以我需要整合这些信息。 关键点: - == 运算符:在Java中,==用于基本数据类型(如int, char)时比较值;对于对象引用时比较内存地址(即是否指向同一个对象)。 - equals() 方法:这是Object类的一个方法,默认实现是使用==比较内存地址。但许多类(如String重写equals()来比较内容或值。 根据引用[3][4],对于字符串,==比较地址,equals()比较内容。引用[4]还强调,equals()可以比较对象的属性值,如果被重写。 我的回答需要: - 使用中文,因为系统指令说“使用中文回答”。 - 结构清晰,逐步解释。 - 包括行内数学表达式用$...$,但在这个上下文中,可能不需要数学表达式,除非有相关计算。用户问题不涉及数学,所以可能不需要。 - 独立公式用$$...$$,但同样,这里可能不适用。 - 生成相关问题:在回答末尾添加相关问题。 - 在引用的段落末尾添加引用标识,如[^1]。 系统指令说:“回答中引用的段落末尾自然地添加引用标识”。所以,当我引用信息时,需要添加引用。 参考的引用是[^1]到[^4],所以我应该使用这些来支持我的回答。 回答结构: 1. 介绍:简述==equals()的基本区别。 2. 详细解释==运算符:用于基本类型对象引用。 3. 详细解释equals()方法:默认行为重写情况。 4. 示例:使用提供的引用中的例子或创建新例子。 5. 总结区别。 6. 添加相关问题。 确保回答真实可靠,基于提供的引用。 逐步构建回答: - 开头:直接回答用户问题。 - 解释==:当用于基本数据类型时,比较值;当用于对象时,比较内存地址。引用[1][3]支持。 - 解释equals():默认比较内存地址(与==相同),但许多类如String重写它以比较内容。引用[3][4]支持。 - 示例:使用字符串例子,如引用[3]中的:new String("hello") == new String("hello") 为false,但equals()为true。 - 注意事项:引用[4]提到重写equals()时需要考虑对象一致性。 - 相关问题:生成3-4个相关问题。 相关问题想法: 1. 如何重写equals()方法? 2. 在哪些场景下应该使用==而不是equals()? 3. equals()hashCode()的关系是什么? 4. 对于基本数据类型包装类(如Integer),==equals()的行为如何? 最后,添加引用标识。</think>### Java中'==''equals()'的区别Java中,运算符`==`方法`equals()`都用于比较操作,但它们的用途行为有本质区别。理解这些区别对于避免常见编程错误(如字符串或对象比较不准确)至关重要。下面我将逐步解释两者的差异,并结合示例说明。 #### 1. **`==`运算符的基本行为** - **作用**:`==`是Java中的关系运算符,用于比较两个操作数的“值”。但具体行为取决于操作数的类型: - **对于基本数据类型**(如`int`, `char`, `double`等),`==`直接比较变量的值是否相等。例如: ```java int a = 1; int b = 2; System.out.println(a == b); // 输出 false,因为值不相等 [^1] ``` - **对于对象引用**(如`String`, 自定义类等),`==`比较的是内存地址(即对象在堆中的引用地址)。如果两个引用指向同一个对象实例,则返回`true`;否则返回`false`,无论对象内容是否相同。例如: ```java String s1 = "哪吒"; String s2 = new String("哪吒"); // 通过new创建新对象 System.out.println(s1 == s2); // 输出 false,因为s1s2是不同的对象引用 [^3] ``` - **关键点**:`==`不关心对象的内容,只关心引用是否指向同一块内存。 #### 2. **`equals()`方法的基本行为** - **作用**:`equals()`是`Object`类中定义的方法(所有Java类都继承自`Object`),默认实现是比较对象的内存地址(即与`==`行为相同)。但许多内置类(如`String`)重写了`equals()`方法,使其比较对象的“内容”或“值”: - **默认行为**:如果不重写,`equals()`等价于`==`,比较内存地址。 - **重写后的行为**:对于`String`、`Integer`等类,`equals()`被重写为比较对象的内容是否相等。例如: ```java String name1 = "哪吒"; String name2 = new String("哪吒"); System.out.println(name1.equals(name2)); // 输出 true,因为内容相同 [^3] ``` - 对于自定义类,开发者可以重写`equals()`来定义“相等性”标准(如比较属性值)。引用[4]指出:重写时需先检查是否为同一对象(`==`),然后比较属性值;如果所有属性相等,则返回`true`,否则`false` [^4]。 - **关键点**:`equals()`专注于逻辑相等性,而非物理地址。它更适合比较对象的内容。 #### 3. **核心区别总结** - **比较维度**: - `==`:比较值(基本类型)或内存地址(引用类型)。 - `equals()`:比较内容或自定义逻辑(如果重写)。 - **适用场景**: - 使用`==`当需要检查基本数据类型的值相等或对象引用是否相同(如单例模式)。 - 使用`equals()`当需要比较对象的内容(如字符串、集合元素)。 - **常见误区**:在字符串比较中,误用`==`可能导致错误。例如: ```java String str1 = "hello"; String str2 = new String("hello"); System.out.println(str1 == str2); // false(地址不同) System.out.println(str1.equals(str2)); // true(内容相同) [^3] ``` #### 4. **示例代码演示** 以下代码综合展示`==``equals()`的行为: ```java public class ComparisonDemo { public static void main(String[] args) { // 基本数据类型比较 int x = 5; int y = 5; System.out.println(x == y); // true(值相等) [^1] // 对象引用比较 String obj1 = new String("Java"); String obj2 = new String("Java"); System.out.println(obj1 == obj2); // false(不同地址) [^3] System.out.println(obj1.equals(obj2)); // true(内容相同) [^3] // 三元运算符结合equals()的实用示例 String name = "哪吒"; boolean ret = name.equals("哪吒") ? true : false; // 简化比较 System.out.println(name + (ret ? "很强" : "很弱")); // 输出 "哪吒很强" [^2] } } ``` #### 5. **注意事项** - 重写`equals()`时,必须同时重写`hashCode()`方法,以保持一致性(例如在`HashMap`中使用)[^4]。 - 对于基本数据类型的包装类(如`Integer`),`==`可能因缓存机制(如`-128`到`127`)而表现不同,但`equals()`总是比较值。 - 在条件判断中,优先使用`equals()`避免空指针异常(如`"text".equals(variable)`而非`variable.equals("text")`)。 通过以上分析,您可以看到`==``equals()`在比较机制上的根本差异:`==`关注地址或值,`equals()`关注逻辑内容。正确选择能提升代码健壮性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值