一、前言
示例代码:
public static void main(String[] args) throws IOException {String str1 = new String("hello");String str2 = new String("hello");String str3 = "cde";String str4 = "cde";int i1 = 3;int i2 = 3;Integer i3 = new Integer(4);Integer i4 = new Integer(4);System.out.println(str1 == str2); //falseSystem.out.println(str1.equals(str2)); //trueSystem.out.println(str3 == str4); //trueSystem.out.println(str3.equals(str4)); //trueSystem.out.println(i1 == i2); //trueSystem.out.println(i3 == i4); //false}
返回结果:
false
true
true
true
true
false
为什么str1与str2输出的结果不一样而str3与str4输出的结果一样,i1与i2输出结果一样而i3与i4输出结果不一样呢?==和equals方法之间的区别是什么?如果在初学Java的时候这个问题不弄清楚,就会导致自己在以后编写代码时出现一些低级的错误。今天就来一起了解一下==和equals方法的区别之处。
二、区别
i1==i2结果为true,这个很容易理解,变量i1和变量i2存储的值都为3,肯定是相等的。而为什么str1和str2两次比较的结果不同?要理解这个其实只需要理解基本数据类型变量和非基本数据类型变量的区别。
在Java中有8种基本数据类型:
整型:byte(1 byte), short(2 byte), int(4 byte) , long(8 byte)
浮点型:float(4 byte), double(8 byte)
字符型: char(2 byte)
布尔型: boolean(JVM规范没有明确规定其所占的空间大小,仅规定其只能够取字面值”true”和”false”)
对于这8种基本数据类型的变量,变量直接存储的是“值”,因此在用关系操作符==来进行比较时,比较的就是 “值” 本身。要注意浮点型和整型都是有符号类型的,而char是无符号类型的(char类型取值范围为0~2^16-1)。
也就是说比如:
int i1=3;int i2=3;
变量i1和变量i2都是直接存储的”3”这个数值,所以用==比较的时候结果是true。
而对于非基本数据类型的变量,在一些书籍中称作为 引用类型的变量。比如上面的str1就是引用类型的变量,引用类型的变量存储的并不是 “值”本身,而是于其关联的对象在内存中的地址。比如下面这行代码:
String str1;
这句话声明了一个引用类型的变量,此时它并没有和任何对象关联。
而 通过new String(“hello”)来产生一个对象(也称作为类String的一个实例),并将这个对象和str1进行绑定:
str1= new String("hello");
那么str1指向了一个对象(很多地方也把str1称作为对象的引用),此时变量str1中存储的是它指向的对象在内存中的存储地址,并不是“值”本身,也就是说并不是直接存储的字符串”hello”。这里面的引用和C/C++中的指针很类似。
因此在用==对str1和str2进行比较时,得到的结果是false。因此它们分别指向的是不同的对象,也就是说它们实际存储的内存地址不同。
equals方法是基类Object中的方法,因此对于所有的继承于Object的类都会有该方法。为了更直观地理解equals方法的作用,直接看Object类中equals方法的实现。
下面是Object类中equals方法的实现:
public boolean equals(Object obj) {return (this == obj);
}
很显然,在Object类中,equals方法是用来比较两个对象的引用是否相等,即是否指向同一个对象。
但是有些朋友又会有疑问了,为什么str1与str2的equals比较输出结果是true?
要知道究竟,可以看一下String类的equals方法的具体实现。
下面是String类中equals方法的具体实现:
@Override public boolean equals(Object other) {if (other == this) {return true;}if (other instanceof String) {String s = (String)other;int count = this.count;if (s.count != count) {return false;}// TODO: we want to avoid many boundchecks in the loop below// for long Strings until we have array equality intrinsic.// Bad benchmarks just push .equals without first getting a// hashCode hit (unlike real world use in a Hashtable). Filter// out these long strings here. When we get the array equality// intrinsic then remove this use of hashCode.if (hashCode() != s.hashCode()) {return false;}char[] value1 = value;int offset1 = offset;char[] value2 = s.value;int offset2 = s.offset;for (int end = offset1 + count; offset1 < end; ) {if (value1[offset1] != value2[offset2]) {return false;}offset1++;offset2++;}return true;} else {return false;}}
可以看出,String类对equals方法进行了重写,用来比较指向的字符串对象所存储的字符串是否相等。
其他的一些类诸如Double,Date,Integer等,都对equals方法进行了重写用来比较指向的对象所存储的内容是否相等。
到此还没有结束,在示例代码中还有str3和str4,这两个对象的==和equals比较结果都返回true,这与str1与str2的情况不符,为什么呢?这里涉及到String的缓冲池问题。
分析如下:
String作为一个对象来使用
例子一:对象不同,内容相同,”==”返回false,equals返回true
String s1 = new String("java");
String s2 = new String("java");System.out.println(s1==s2); //false
System.out.println(s1.equals(s2)); //true
运行结果:
false
true
例子二:同一对象,”==”和equals结果相同
String s1 = new String("java");
String s2 = s1;System.out.println(s1==s2); //true
System.out.println(s1.equals(s2)); //true
运行结果:
true
true
String作为一个基本类型来使用
如果值不相同,对象就不相同,所以”==” 和equals结果一样
String s1 = "java";
String s2 = "java";System.out.println(s1==s2); //true
System.out.println(s1.equals(s2)); //true
运行结果:
true
true
如果String缓冲池内不存在与其指定值相同的String对象,那么此时虚拟机将为此创建新的String对象,并存放在String缓冲池内。
如果String缓冲池内存在与其指定值相同的String对象,那么此时虚拟机将不为此创建新的String对象,而直接返回已存在的String对象的引用。
想了解String缓冲池的,可以看看这篇文章:Java提高篇 —— String缓冲池
三、总结
对于==而言
如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等
如果作用于引用类型的变量,则比较的是所指向的对象的地址
对于equals而言
equals方法不能作用于基本数据类型的变量
如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址
诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容