java 中“==”与“equal” 的区别

一、java当中的数据类型和“==”的含义:

 
基本数据类型(也称原始数据类型) :byte,short,char,int,long,float,double,boolean。他们之间的比较,应用双等号(==),比较的是他们的值。
复合数据类型(类):当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址(确切的说,是堆内存地址)。
注:对于第二种类型,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。因为每new一次,都会重新开辟堆内存空间。

二、equals()方法介绍:
 
JAVA当中所有的类都是继承于Object这个超类的,在Object类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地址,但在一些类库当中这个方法被复写了,如String、Integer、Date。在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。 
所以说,对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是内存中的存放位置的地址值,跟双等号(==)的结果相同;如果被复写,按照复写的要求来。
 
 
 
三、String类的equals()方法:
 
现在我们拿String类来举例:
 
我们去\src\java\lang目录中找到String类,发现equals方法被复写如下:
 
 1     public boolean equals(Object anObject) {
 2         if (this == anObject) {
 3             return true;
 4         }
 5         if (anObject instanceof String) {
 6             String anotherString = (String)anObject;
 7             int n = value.length;
 8             if (n == anotherString.value.length) {
 9                 char v1[] = value;
10                 char v2[] = anotherString.value;
11                 int i = 0;
12                 while (n-- != 0) {
13                     if (v1[i] != v2[i])
14                         return false;
15                     i++;
16                 }
17                 return true;
18             }
19         }
20         return false;
21     }

上述代码可以看出,String类中被复写的equals()方法其实是比较两个字符串的内容。下面我们通过实际代码来看看String类的比较。
 
1、举例代码如下:
 

1 public class StringDemo {
2     public static void main(String[] args) {
3         String s1 = "Hello";
4         String s2 = "Hello";
5         System.out.println(s1 == s2);   // true
6     }
7 }

上方代码中,用“==”比较s1和s2,返回的结果是true。
 
2、稍微改动一下程序,会有奇怪的发现:
 

 1 public class StringDemo {
 2     public static void main(String args[]) {
 3         String str1 = "Hello";
 4         String str2 = new String("Hello");
 5         String str3 = str2; // 引用传递
 6         System.out.println(str1 == str2); // false
 7         System.out.println(str1 == str3); // false
 8         System.out.println(str2 == str3); // true
 9         System.out.println(str1.equals(str2)); // true
10         System.out.println(str1.equals(str3)); // true
11         System.out.println(str2.equals(str3)); // true
12     }
13 }

 上方第4行代码中,我们new了一个对象,用“==”比较s1和s2,返回的结果却是false;而用用“equals”比较s1和s2,返回的结果是true。 
为了分析上面的代码,我们必须首先分析堆内存空间和栈内存空间,这一点非常重要.
 
 
面试题:请解释字符串比较之中“==”和equals()的区别?
 
 ==:比较的是两个字符串内存地址的数值是否相等,属于数值比较;
 equals():比较的是两个字符串的内容,属于内容比较。
以后进行字符串相等判断的时候都使用equals()。
 
3、再次更改程序:
 

1 public class ObjectDemo{
2     public static void main(String[] args) {
3         String s1 = "Hello";
4         String s2 = new String("Hello");
5         s2 = s2.intern();
6         System.out.println(s1 == s2);       //  true
7         System.out.println(s1.equals(s2));  //  true
8     }
9 }

上述代码的第5行中,java.lang.String的intern()方法"abc".intern()方法的返回值还是字符串"abc",表面上看起来好像这个方法没什么用处。但实际上,它做了个小动作:检查字符串池里是否存在"abc"这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会 把"abc"添加到字符串池中,然后再返回它的引用。
 
 
 
四、比较两个对象的值:
 
代码如下:
 
 1 public class ObjectDemo {
 2     public static void main(String args[]){
 3         Student student1 = new Student("生命壹号",22,"成都");
 4         Student student2 = new Student("生命壹号",22,"成都"); 
 5         System.out.println(student1==student2);
 6         System.out.println(student1.equals(student2));
 7     }
 8 }
 9 class Student {
10     private String name;
11     private int age;
12     private String address;
13     public Student(String name,int age,String address){
14         this.name = name;
15         this.age = age;
16         this.address = address;
17     }
18     //重写Object类中的equals方法(比较两个对象的值是否相等)
19     public boolean equals(Object obj){
20         //如果内存地址相等,那么一定是同一个对象,就无需比较两个对象的属性值
21         if(this==obj){
22             return true;
23         }
24         //判断obj是否为Baboon类型的实例
25         if(obj instanceof Student){
26             Student b = (Student)obj;//强制转换
27             //判断两个对象的属性值是否相等
28             if(!this.name.equals(b.name)){
29                 return false;
30             }
31             else if(this.age!=b.age){
32                 return false;
33             }
34             else if(this.address!=b.address){
35                 return false;
36             }
37             return true;
38         }else{
39             return false;
40         }
41     }
42 }

上述代码中,首先判断传递进来的对象与当前对象的地址是否相等,如果相等,则肯定是同一个对象。因为传递进来的参数是Object类型,所以任何对象都可以接收。一旦接收进来,就将Object类型的对象向下转型,然后再做判断。

 

 

 

 

 

 

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); 
    Boolean result3=(t3==t4); 
     
    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); 
    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; 
    Boolean b3=(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 

总之: 
“==”比较的是值【变量(栈)内存中存放的对象的(堆)内存地址】 
equal用于比较两个对象的值是否相同【不是比地址】 

【特别注意】Object类中的equals方法和“==”是一样的,没有区别,而String类,Integer类等等一些类,是重写了equals方法,才使得equals和“==不同”,所以,当自己创建类时,自动继承了Object的equals方法,要想实现不同的等于比较,必须重写equals方法。

"=="比"equal"运行速度快,因为"=="只是比较引用.

posted on 2017-10-19 11:41  DiscretOrz  阅读(149)  评论(0编辑  收藏  举报

导航