不知道是不是很多人和我一样,一直都以为==比较的是引用对象的地址,equals比较的是对象的值?
先来做个实验:
Test5 test3 = new Test5("spot", "Ruff!");
Test5 test4 = new Test5("spot", "Ruff!");
System.out.println("test3 == test4:" + (test3 == test4));
System.out.println("test3.equals(test4):" + (test3.equals(test4)));
如果单纯的认为equals是比较对象的值,那么结果就是:false,true?
但事实上,结果为:false,false
那equals和==的区别在哪呢?
其实在大多数类中,==和equals是没有区别的。
所有的对象都继承于Object,而Object中的equals方法的返回结果,就是调用==比较的值。只有部分类会自己重写equals方法,如:String,Integer等。
而他们产生区别的原因,恰恰就是因为重写了equals方法,具体差异根据具体重写后的代码而异。
先看下Integer类中,==和equals的区别:
Integer n1 = new Integer(12);
Integer n2 = new Integer(12);
System.out.println("n1 == n2:" + (n1 == n2));
System.out.println("n1.equals(n2):" + (n1.equals(n2)));
结果为:
n1 == n2:false
n1.equals(n2):true
可以看到,Integer的equals方法是比较对象的值,而Integer的equals源码如下:
public boolean equals(Object obj) {
// 判断obj对象是否是
if (obj instanceof Integer) {
// 比较对象obj的值和当前对象的值是否是同一引用
return value == ((Integer)obj).intValue();
}
return false;
}
我们知道,new出来的对象是存储在堆中,而new Integer(12)中的12是存储在常量池中。这也就是为什么==返回的结果是false,而equals返回的结果是true了。
因为==比较的是堆中的new对象,而equals比较的是常量池中的数值,new是会一直在堆中创建新的数据,而常量池中的数据如果存在,就会直接引用,不会重新创建。
String的情形其实和Integer类很相似:
// 在常量池中直接引用
String a = "hello";
String b = "hello";
System.out.println("a==b:" + (a==b));
System.out.println("a.equals(b):" + a.equals(b));
// 在堆中创建对象
String aa = new String("hello");
String bb = new String("hello");
System.out.println("aa==bb:" + (aa==bb));
System.out.println("aa.equals(bb):" + aa.equals(bb));
返回的结果为:
a==b:true
a.equals(b):true
aa==bb:false
aa.equals(bb):true
可以看到,对象直接在常量池中进行引用,==和equals的结果其实是一样的,因为比较的都是常量池中对象的比较;
而在堆中进行创建对象,就会出现==是比较堆中对象的引用,而equals是比较对常量池中对象的引用。
如果对上面讲的堆和常量池的存储过程有不懂的,我之前有篇博客介绍了JVM中变量和值的存储位置,大家可以去看下。