• Java语法----Java中equals和==的区别(转载)


    【声明】

    欢迎转载,但请保留文章原始出处→_→

    生命壹号:http://www.cnblogs.com/smyhvae/

    文章来源:http://www.cnblogs.com/smyhvae/p/3929585.html

    【正文】

    平时在学Android和Java语言的时候,总是碰到“equals”和“==”这两个字符,老感觉差不多;其实还是有一些区别的,今天干脆把它们彻底弄清楚。

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

    • 基本数据类型(也称原始数据类型) :byte,short,char,int,long,float,double,boolean。他们之间的比较,应用双等号(==),比较的是他们的值
    • 引用数据类型:当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址(确切的说,是堆内存地址)。

    注:对于第二种类型,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。因为每new一次,都会重新开辟堆内存空间

    二、equals()方法介绍:

    JAVA当中所有的类都是继承于Object这个超类的,在Object类中定义了一个equals的方法,equals的源码是这样写的:

    public boolean equals(Object obj) {
        //this - s1
        //obj - s2
        return (this == obj);
    }

    可以看到,这个方法的初始默认行为是比较对象的内存地址值,一般来说,意义不大。所以,在一些类库当中这个方法被重写了,如String、Integer、Date。在这些类当中equals有其自身的实现(一般都是用来比较对象的成员变量值是否相同),而不再是比较类在堆内存中的存放地址了。 
    所以说,对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是内存中的存放位置的地址值,跟双等号(==)的结果相同;如果被复写,按照复写的要求来。

    我们对上面的两段内容做个总结吧:

     == 的作用:
      基本类型:比较的就是值是否相同
      引用类型:比较的就是地址值是否相同
    equals 的作用:
      引用类型:默认情况下,比较的是地址值。
    注:不过,我们可以根据情况自己重写该方法。一般重写都是自动生成,比较对象的成员变量值是否相同

    三、String类的equals()方法:

    现在我们拿String类来举例:

    我们去srcjavalang目录中找到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 package com.smyh;
     2 
     3 public class ObjectDemo {
     4     public static void main(String args[]){
     5         Student student1 = new Student("生命壹号",22,"成都");
     6         Student student2 = new Student("生命壹号",22,"成都"); 
     7         System.out.println(student1==student2);
     8         System.out.println(student1.equals(student2));
     9     }
    10 }
    11 class Student {
    12     private String name;
    13     private int age;
    14     private String address;
    15     public Student(String name,int age,String address){
    16         this.name = name;
    17         this.age = age;
    18         this.address = address;
    19     }
    20     //重写Object类中的equals方法(比较两个对象的值是否相等)
    21     public boolean equals(Object obj){
    22         //为了提高效率:如果两个内存地址相等,那么一定是指向同一个对内存中的对象,就无需比较两个对象的属性值(自己跟自己比,没啥意义嘛)
    23         if(this==obj){
    24             return true;
    25         }
    26         
    27         //为了提供程序的健壮性
    28         //我先判断一下,obj是不是学生的一个对象,如果是,再做向下转型,如果不是,直接返回false。
    29         //这个时候,我们要判断的是对象是否是某个类的对象?
    30         //记住一个格式:对象名 instanceof 类名。表示:判断该对象是否是该类的一个对象       
    31         if(!(obj instanceof Student)){        
    32             return false;                
    33         }
    34         
    35         //如果是就继续
    36         Student s = (Student)obj;//强制转换,即向下转型(毕竟Object类型没有具体的对象属性)        
    37         return this.name.equals(s.name) && this.age == s.age && this.address.equals(s.address);//判断两个对象的属性值是否相等
    38     }
    39 }
    复制代码

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

     运行效果:

    其实,如果是在Eclipse中做开发的话,上面重写的equals()方法其实是可以自动生成的:

    自动生成后的equals()方法如下:

    复制代码
     1     @Override
     2     public boolean equals(Object obj) {
     3         if (this == obj)
     4             return true;
     5         if (obj == null)
     6             return false;
     7         if (getClass() != obj.getClass())
     8             return false;
     9         Student other = (Student) obj;
    10         if (address == null) {
    11             if (other.address != null)
    12                 return false;
    13         } else if (!address.equals(other.address))
    14             return false;
    15         if (age != other.age)
    16             return false;
    17         if (name == null) {
    18             if (other.name != null)
    19                 return false;
    20         } else if (!name.equals(other.name))
    21             return false;
    22         return true;
    23     }
    复制代码

    可以看到:我们自己重写的equals()方法和自动生成的equals()方法,二者效果是一样的。

    【参考】

    参考博客:

    http://www.cnblogs.com/zhxhdean/archive/2011/03/25/1995431.html

    http://www.cnblogs.com/zhxhdean/archive/2011/03/26/1996468.html

    以及李兴华老师的视频讲解。

  • 相关阅读:
    常见的web漏洞
    WEB前端性能优化常见方法
    前端多终端浏览器兼容
    vue中computer与watch区别
    Vue父子组件生命周期执行顺序
    Git配置文件的妙用
    Git的基本概念和操作
    对AUC计算公式和几何意义的理解(详细版)
    如何看文献
    Python函数 range()和arange()的区分
  • 原文地址:https://www.cnblogs.com/May-study/p/9481351.html
Copyright © 2020-2023  润新知