区别
==:比较的是对象的地址,也就是比较是否是同一个对象
equals: 比较的是对象的值
原理
Java中equals和==的区别 java中的数据类型,可分为两类:
- 基本数据类型,也称原始数据类型。
byte,short,char,int,long,float,double,boolean
他们之间的比较,应用双等号(==),比较的是他们的值。
public void method1(){
double i = 235;
double j = 235;
System.out.println(i==j);
System.out.println(new Double(i).equals(j));}
解析:
输出结果为
true
true
因为是原始数据类型,所以==比较的是原始数据类型的值,对于equals我们来看一下源码的解析:
public boolean equals(Object obj) {
return (obj instanceof Double)&& (doubleToLongBits(((Double)obj).value)==\
doubleToLongBits(value));
}
可以看出来源码中调用的仍然是 = = 方法来完成equals比较。所以这里返回true。
复合数据类型(类)
- Object(源码分析)
public boolean equals(Object obj) {
return (this == obj);
}
可以看出在Object对象中,调用equals方法中默认调用的是==方法,也就是比较的是是否为同一对象。
- 比较不同对象
public void method1(){
Person p1 = new Person(12,"wwj");
Person p2 = new Person(24,"dx");
System.out.println(p1==p2);
System.out.println(p1.equals(p2));}
解析:
输出结果为:
false
false
- 比较相同对象
public void method1(){
Person p1 = new Person(12,"wwj");
Person p2 = p1;
System.out.println(p1==p2);
System.out.println(p1.equals(p2));}
解析:
输出结果为:
true
true
总结:
上面的两种情况验证了源码中equals调用的 = = 方法。当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。
JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地址,但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。
String内存中equals方法的实现:
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;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
由源码可知:String对象的equals的对象比较的是字符串中的字符是否相等。
对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是基于他们在内存中的存放位置的地址值的,因为Object的equals方法也是用双等号 = = 进行比较的,所以比较后的结果跟双等号 = = 的结果相同。