为什么Java中不能用==比较对象内容_==与equals差异解析

==比较内存地址,equals()比较内容。String等类重写equals()后可比较逻辑相等性,自定义类需重写equals()和hashCode(),避免null调用,包装类型慎用==。

在Java中,==equals() 都可以用来判断“相等”,但它们的行为完全不同。很多人初学时误以为 == 能比较对象内容,结果导致逻辑错误。根本原因在于:== 比较的是内存地址,而 equals() 才是用来比较对象内容的。

== 比较的是引用(内存地址)

当使用 == 比较两个对象时,Java 判断的是这两个变量是否指向堆内存中的同一个对象实例。

例如:

String str1 = new String("hello");
String str2 = new String("hello");
System.out.println(str1 == str2); // 输出 false

虽然 str1 和 str2 内容相同,但它们是两个不同的对象,分别占用不同的内存空间,所以 == 返回 false。

equals() 比较的是对象内容

equals() 是 Object 类的方法,设计初衷就是用于比较两个对象的“逻辑相等性”。但默认实现仍是 ==,因此需要类自己重写 equals() 方法才能正确比较内容。

像 String、Integer 等常用类都已经重写了 equals():

继续上面的例子:

System.out.println(str1.equals(str2)); // 输出 true

因为 String 类重写了 equals(),它会逐个字符比较内容,所以返回 true。

常见误区与注意事项

以下几点容易被忽略:

  • 自定义类如果不重写 equals(),即使内容一样,equals 也会返回 false(因为默认用 == 比较)
  • 基本数据类型(如 int、char)可以用 == 安全比较值,因为不是对象
  • 包装类型(如 Integer)用 == 有风险:-128 到 127 的值会被缓

    存,可能“碰巧”相等
  • 比较前务必确认对象非 null,否则调用 equals() 会抛 NullPointerException

正确的做法是:总是用 equals() 比较对象内容,并确保相关类正确实现了该方法。如果自己写类,记得同时重写 equals() 和 hashCode(),以保证一致性。

基本上就这些——记住:== 看“是不是同一个东西”,equals() 看“是不是一样的东西”。