equals()与 == 在java编程或系统实际可发中都会用到,但是二者又有实际的区别。千万不要理所当然地认为equals()与==的作用就是一样的,下面就一步步来阐述二者的区别。
phone.java
新建phone类,有两个私有的成员变量price与memory,在构造器中对其进行初始化。
package testPackage;
public class phone {
private float price;//价格
private int memory;//内存
public phone(float price, int memory) {
super();
this.price = price;
this.memory = memory;
}
}
testPhone.java
新建两个手机对象,两个私有成员变量都赋一样值,用equals()进行比较
package testPackage;
public class testPhone {
public static void main(String[] args){
phone p1 = new phone(4000f,16);//4000元,16G内存的手机p1
phone p2 = new phone(4000f,16);//4000元,16G内存的手机p2
if(p1.equals(p2)){//如果p1等于p2
System.out.println("p1与p2相同");
}
else{//p1不等于p2
System.out.println("p1与p2不相同");
}
}
}
运行结果:
p1与p2不相同
结果分析:
尽管两个手机对象的私有成员变量全部相同,equals()运行结果显示这两个对象是不相同的。这是为什么呢?
还有,phone类中并没有定义equals()方法,这个equals()方法哪里来的?
相信大家都知道java中的所有类都继承了Object.class,因此,所有类都继承了Object类中的equals()方法,要想知道运行结果的本质原因必须查看Object.class的源码。
打开Object.class的源码找到定义equals()方法的位置:
public boolean equals(Object obj) {
return (this == obj);
}
发现Object.class给出equals()方法的定义,竟然只是简单地通过==去实现判断。而==在java中用于判断存储在栈中的值是否相等。我们的基本数据类型像int的比较都用==,因为基本数据类型就存放在栈中,同样存放在栈中的,还有引用数据类型的引用,而真正的对象本身则存放在堆中。因此,这里给出的equals()方法就只是比较两个对象在内存中的引用(即地址)是否相等。本例中新建了两个对象,在内存中就新开辟了两个内存空间,引用地址当然是不一样的,程序运行结果当然是不相同。
但是我们想通过两个手机的价格与内存相同就判断手机是相同的,确实实际开发中我们都是这样想的,那应该怎么办呢?重写equals()方法。
phone.java中重写equals()方法:
package testPackage;
public class phone {
private float price;//价格
private int memory;//内存
public phone(float price, int memory) {
super();
this.price = price;
this.memory = memory;
}
@Override
public boolean equals(Object obj){
if(this == obj){//如果引用地址是相同的,证明是同一个对象,那当然相同
return true;
}
if(obj ==null){//如果对象是空的,没有比较的必要,直接返回false
return false;
}
if(!(obj instanceof phone)){//如果obj不是phone的一个对象实例,直接返回false
return false;
}
phone p = (phone)obj;//编译需要,强制类型转换
if( this.price==p.price && this.memory==p.memory ){
return true;
}
else{
return false;
}
}
}
重新运行testPhone.java,查看运行结果:
运行结果:
p1与p2相同
结果分析:
通过重写equals()实现了通过私有成员变量判断两个手机对象是否相等,若prize与memory都相等,则两个手机对象就相等。
看到这里,相信大家对equals()与==的区别都有一定理解了,但是显然还不够,关于二者的区别,还有值得了解的地方。
新建testString.java进行字符串测试
package testPackage;
public class testString {
public static void main(String[] args){
String s1 = "我是手机1";
String s2 = new String("我是手机1");
System.out.println(s1==s2);
System.out.println(s1.equals(s2));
}
}
运行结果:
false
true
结果分析:
在java中,字符串是引用数据类型,以实例对象的形式存在。因此,虽然s1与s2看上去定义方式不同,但是它们都新建了对象,本质其实是一样的。
如上所说,==比较的是引用数据类型在栈中的地址是否相同,s1与s2都新建了两个字符串对象,当然开辟了两个不同的内存空间,结果返回false当然是意料之中的。
但是s1.equals(s2)为什么返回true呢?按上面所说Object.class不也是直接通过==进行判断的吗?结果应该是false吧?这时不妨大胆打开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;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
发现String.class给出了对Object.class中equals()方法的重写,重写后equals()要判断两个字符串的字符是否全部相等。
这是修改一下s2的字符再重新测试:
public static void main(String[] args){
String s1 = "我是手机1";
String s2 = new String("我是手机2");
System.out.println(s1==s2);
System.out.println(s1.equals(s2));
}
运行结果:
false
false
结果分析:
由于s1与s2的字符不完全相同,因此equals()方法返回false
到这里,给出在实际应用中equals()与==的使用技巧:
- 若是两个基本数据类型的比较,例如int a = 1,b = 2;比较a是否等于b。使用==
- 若要通过比较成员变量判断两个对象是否相等(对象不是字符串),例如自己写的phone对象,要重写从Object类继承来的equals()方法实现。
- 若是比较两个字符串,直接用equals()方法。