equals 与 ‘==’ 的本质区别在笔者看来,equals与 运算符最本质的区别在于:equals是一个方法,对象名.equals(参数)进行使用; 是像 +、-、* 这类的运算符,直接使用。2.对象比较的区别 结合一段代码分析
A a=new A();
A a1=new A(); //创建两个A类的对象 a,a1
System.out.println((a==a1)); //用==比较对象a和a1
System.out.println((a.equals(a1))); //用equals 比较a和a1
运行结果:false false我们要知道对于 == 来说比较两个对象时,实质上是检查对象的内存地址是否相等,由于这里使用了new关键字,这意味着对于对象a和a1,它们的内存地址是不一样的,返回值为false。我们可以用以下语句将其地址输出:
System.out.println(a);
System.out.println(a1);
运行结果:Tech715.Data716
A
@
15
d
b
9742
T
e
c
h
715.
D
a
t
a
716
A@15db9742 Tech715.Data716
A@15db9742Tech715.Data716A@6d06d69c
格式为:a.getClass().getName()+"@"+Integer.toHexString(a.hashCode())对于equals 来说,结合系统的源码:public boolean equals(Object obj) { return (this == obj); }
此方法是用来检查对象是否是同一个,显然a和a1不是同一个对象,因此返回值为false。3.字符串的比较在比较基本类型的两个变量是否相等,我们会使用==运算符,但这个运算符不适用于比较String对象,为此我们结合代码分析其原因。
//首先定义4个String类的变量
String str1 = "hello";
String str2 = " world";
String str3 = "hello world";
str1 += str2; //让str1的字符串内容与str3的内容相等
System.out.println("str1 is: " + str1);
System.out.println(“str3 is: " + str3);
System.out.println(”: "+(str1 == str3));
System.out.println("equals: "+str1.equals(str3));
运行结果:str1 is: hello worldstr3 is: hello world: falseequals: true
对于 str1 == str3 表达式,运算符进行比较时,将检查这两个String变量是否引用了同一个字符串,如果它们引用的是不同的字符串,其返回值为false,而不管其字符串内容是否相等。在这里,str1 和str3 引用的是分开在两处的两个字符串,返回false。若让 str3 = str1,则此时str1和str3引用同一个字符串,返回true。但是不论哪一种情况对于equals来说只要字符串相等其返回值都为true,这也是为什么我们在比较字符串时首选的方法是用equals来比较。
那么有没有一种方法能让运算符适用于字符串的比较呢?答案是有的,用字符串扣留机制能实现 == 与equals产生相同的比较结果。字符串扣留功能确保了两个String对象不能封装同一个字符串,因此所有的String对象均封装惟一的字符串。这就意味着,如果两个String变量引用的字符串相等,那么这两个引用也必定相等。换一种说法就是,如果两个String变量包含不相等的引用,它们引用的字符串也必定不相等。如何安排才能使得所有的String对象封装惟一的字符串呢?这只需要为每一个String对象调用intern方法。还是上面的例子,在str1 += str2;后加上
str1=str1.intern();
str3=str3.intern();
最后str1 == str3;表达式返回结果为true。这就是字符扣留的功能实现,它能减少用户程序中存储String对象需要的存储量,实现== 替代equals()方法。