深入理解equals与==的区别

来源:互联网 发布:mac 输出当前路径 编辑:程序博客网 时间:2024/05/16 07:38

区别:

  1. “==”比较的是变量引用的对象内存地址是否是同一个地址,即是否是同一对象【变量(在栈中)内存中存放的对象的(在堆中)内存地址】。
  2. equal用于比较两个对象的值是否相同【往往是比成员变量值不是比地址】。

下面结合代码来说明:

public class EqualTest { public static void main(String[] args) {     //对于基本类型的变量。"=="和"equal"的区别     int t1=57;     int t2=67;     int t3=124;     int t4=124;     //“==”对于基本数据类型,实质还是判断两个变量的引用的内存地址是否相等。     Boolean result1=(t1==t2);     Boolean result2=((t1+t2)==t3); //true,因t1+t2的值在栈中与t3的为同一个,他们的地址是相同的。    Boolean result3=(t3==t4); //true,因t3的值在栈中与t4的为同一个字面值,VM中字面值存放在栈中是共享的,它们的地址是相同的。    System.out.println("/n/n-----【t1==t2】"+result1+"/n-----【(t1+t2)=t3】"+result2+"/n-----【t3=t4】"+result3);     //“equal”是类的一个方法,所以不能用于基本数据类型,只能用于类变量。对于基本数据类型要用其包装类。     Integer i1=new Integer(t1);     Integer i2=new Integer(t2);     Integer i3=new Integer(t3);     Integer i4=new Integer(t4);     Boolean ri1=i1.equals(i2);     Boolean ri2=i3.equals(i1+i2); //true,因为包装类在进行+-*/运算时会自动进行拆包,转换为基本数据类型参与运算,而equals又是比较值的。    Boolean ri3=i3.equals(i4);     System.out.println("/n/n-----【i1.equals(i2)】"+ri1+"/n-----【i3.equals(i1+i2)】"+ri2+"/n-----【i3.equals(i4)】"+ri3);     //对于对象变量,"=="和"equal"的区别     String st1="wasiker ";     String st2="is super man";     String st3="wasiker is super man";     String st4="wasiker is super man";     Boolean b1=(st1==st2);     Boolean b2=(st1+st2)==st3; //false,因为==比较对象的内存地址,而(st1+st2)运算后被翻译为StringBuilder sb = new StringBuilder(); sb.append(st1); sb.append(st2); String i = sb.toString();实际上是i与st3引用的内存地址比较,显然他们的内存地址是不一样的。     Boolean b3=(st3==st4); //true,字符串"wasiker is super man"在栈中是共享的且不可修改的,vm会重复利用之。当st3、st4创建到"wasiker is super man"的引用时,vm会先在栈中寻找是否存在这样的字符串,如果有就直接将其内存地址赋给变量st3、st4.    System.out.println("/n/n-----【st1==st2】"+b1+"/n-----【(st1+st2)==st3】"+b2+"/n-----【st3==st4】"+b3); //因为对象变量的存储的是对象在内存中的路径,即内存地址。所以用“==”比较时,即使 //对象的值相等,但是他们的内存地址不同,所以==的结果为false。故“==”用于比较两 //个变量的值是否相等,而不是变量引用的对象是否相等     Boolean r1=st1.equals(st2);     Boolean r2=(st1+st2).equals(st3);     Boolean r3=st3.equals(st4);     System.out.println("/n/n-----【st1.equals(st2)】"+r1+"/n-----【(st1+st2).equals(st3)】"+r2+"/n-----【st3.equals(st4)】"+r3); //equal用于比较两个对象是否相同。 } } 

运行结果为:
—–【t1==t2】false
—–【(t1+t2)=t3】true
—–【t3=t4】true

—–【i1.equals(i2)】false
—–【i3.equals(i1+i2)】true
—–【i3.equals(i4)】true

—–【st1==st2】false
—–【(st1+st2)==st3】false
—–【st3==st4】true

—–【st1.equals(st2)】false
—–【(st1+st2).equals(st3)】true
—–【st3.equals(st4)】true

特别注意:

Object类中的equals方法和“==”是一样的,没有区别,由于Object 类是所有类的最高基类,所有其他类都继承类Object类的equals()方法,定义原型如下:

public boolean equals (Object x){    return this == x;}

在java中 “==” 是判断两个对象是否同一,而不是判断相等。因此Object类中的equals方法也是判断两个对象是否同一。而String类,Integer类等等一些类,是重写了equals方法,才使得equals和==不同,所以,当自己创建类时,自动继承了Object的equals方法,要想实现不同的等于比较,必须重写equals方法。