使用==进行比较:
- 比较基础数据类型(Java中基础数据类型包括八中:short,int,long,float,double,char,byte,boolen):这种情况下,==比较的是他们的值是否相等。
- 引用间的比较:在这种情况下,==比较的是他们在内存中的地址,也就是说,除非引用指向的是同一个new出来的对象,此时他们使用
==
去比较得到true,否则,得到false。
使用equals进行比较:
equals是Object类中的一个方法,在该类中,equals的实现也仅仅只是比较两个对象的内存地址是否相等,但在一些子类中,如:String、Integer 等,该方法将被重写。
以String类为例子说明eqauls与==的区别:
String string1 = "me";
String string2 = "me";
System.out.println(string1 == string2);
System.out.println(string1.equals(string2));
String string2 = new String("me");
System.out.println(string1 == string2);
System.out.println(string1.equals(string2));
String string2 = new String("me").intern();
System.out.println(string1 == string2);
System.out.println(string1.equals(string2));
结果是:
true
true
false
true
true
true
在分析结果之前我们需要知道JVM处理String的一些特性。Java的虚拟机在内存中开辟出一块单独的区域,用来存储字符串对象,这块内存区域被称为字符串缓冲池。当使用 String a = "abc"这样的语句进行定义一个引用的时候,首先会在字符串缓冲池中查找是否已经相同的对象,如果存在,那么就直接将这个对象的引用返回给a,如果不存在,则需要新建一个值为"abc"的对象,再将新的引用返回a。String a = new String("abc");这样的语句明确告诉JVM想要产生一个新的String对象,并且值为"abc",于是就在堆内存中的某一个小角落开辟了一个新的String对象。
==在比较引用的情况下,会去比较两个引用的内存地址是否相等。
结果分析:
- 第一个true是因为已经在内存中创建了一个值为“me”的对象了,String2只是获取了“me”的引用即他们的指向的地址是相同的。
- 第二个true是因为
String
已经重写了equals
方法,比较的是具体对象内容是否相等。 - 第三个false显而易见是因为新创建了String2的对象,所以使用“==”来比较引用地址必然是不相等的。
- 第四个true跟第二个true的原因是相同的,他们比较的具体对象的内容而不是内存地址,所以是true。
- 第五个true是因为String2虽然new了一个新的对象,但是又调用了intern方法,将该对象置入常量池中,只是获取了“me”的引用即他们的指向的地址是相同的。
- 第六个true是因为
String
已经重写了equals
方法,比较的是具体对象内容是否相等。
我们也可以通过equals针对String重写的方法的源码来进行学习理解:
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;
//在这里循环遍历两个String中的char
while (n-- != 0) {
//只要有一个不相等,那么就会返回false
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;