深入理解equals与==的区别

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/tyhj_sf/article/details/73771957

区别:

  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方法。

猜你喜欢

转载自blog.csdn.net/tyhj_sf/article/details/73771957