美文网首页
1.浅谈“==”和equal的区别及其引发的思考

1.浅谈“==”和equal的区别及其引发的思考

作者: 文茶君 | 来源:发表于2019-11-08 11:35 被阅读0次

    最近做java基础题遇到好多这种,然后感觉文档也写的不全。由于这种问题出现的几率太高了,所以我就打算也写一写关于这方面的知识,为了提醒自己不犯一些低级失误。

    找了下资料

    浅谈Java中的equals和== - Matrix海子 - 博客园

    Java 基本数据类型 及 == 与 equals 方法的区别 - Latiny - 博客园

    public class Main {

                      public static void main(String[] args) {

                        int n=3;

                        int m=3;

                        System.out.println(n==m);

                        String str = new String("hello");

                        String str1 = new String("hello");

                        String str2 = new String("hello");

                            System.out.println(str1==str2);

                            str1 = str;

                                str2 = str;

                        System.out.println(str1==str2);

        }

    }

    输出结果为 true false true

      n==m结果为true,这个很容易理解,变量n和变量m存储的值都为3,肯定是相等的。

           对于这8种基本数据类型的变量,变量直接存储的是“值”,因此在用关系操作符==来进行比较时,比较的就是 “值” 本身。

    也就是说比如:

      int n=3;

      int m=3;

      变量n和变量m都是直接存储的"3"这个数值,所以用==比较的时候结果是true。

      而对于非基本数据类型的变量,在一些书籍中称作为 引用类型的变量。比如上面的str1就是引用类型的变量,引用类型的变量存储的并不是 “值”本身,而是于其关联的对象在内存中的地址。比如下面这行代码:

      String str1;

      这句话声明了一个引用类型的变量,此时它并没有和任何对象关联。

      而 通过new String("hello")来产生一个对象(也称作为类String的一个实例),并将这个对象和str1进行绑定:

      str1= new String("hello");

      那么str1指向了一个对象(很多地方也把str1称作为对象的引用),此时变量str1中存储的是它指向的对象在内存中的存储地址,并不是“值”本身,也就是说并不是直接存储的字符串"hello"。这里面的引用和C/C++中的指针很类似。

      因此在用==对str1和str2进行第一次比较时,得到的结果是false。因此它们分别指向的是不同的对象,也就是说它们实际存储的内存地址不同。

      而在第二次比较时,都让str1和str2指向了str指向的对象,那么得到的结果毫无疑问是true。


    equals方法是基类Object中的方法,因此对于所有的继承于Object的类都会有该方法。为了更直观地理解equals方法的作用,直接看Object类中equals方法的实现。

    Object中equal的方法

    很显然,在Object类中,equals方法是用来比较两个对象的引用是否相等,即是否指向同一个对象。

      但是有些朋友又会有疑问了,为什么下面一段代码的输出结果是true?

    public class Main {

            public static void main(String[] args) {

            String str1 = new String("hello");

            String str2 = new String("hello");

            System.out.println(str1.equals(str2));

        }

    }

    String中equal()方法

    可以看出,String类对equals方法进行了重写,用来比较指向的字符串对象所存储的字符串是否相等。

      其他的一些类诸如Double,Date,Integer等,都对equals方法进行了重写用来比较指向的对象所存储的内容是否相等。

      总结来说:

      1)对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;

        如果作用于引用类型的变量,则比较的是所指向的对象的地址

      2)对于equals方法,注意:equals方法不能作用于基本数据类型的变量

        如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;

        诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容。


    那么对于数组的equal方法 。数组并没有对equals进行Override,依然是Object的equals 

    Person p1 =newPerson(1001, "latiny1");

    Person p2 =newPerson(1001, "latiny1");

    4System.out.println(p1 == p2);

    System.out.println(p1.equals(p2));

    输出结果:

    false

    false

    这里"=="比较的是内存地址,"equals"比较的也是地址,没有重写equals方法的类都是调用的Object的equals的方法。


    那么接下来是容易错的包装类

    Character a =newCharacter('A');

     Character b =newCharacter('A');

     System.out.println(a == b); 

     System.out.println(a.equals(b)); 

     Integer i1 =newInteger(10); 

    Integer i2 =newInteger(10);

    System.out.println(i1 == i2);//这个容易错

    System.out.println(i1.equals(i2));

    输出结果:

    false

    true

    false

    true

      这边"=="比较的是对象的内存地址,new了两个不同的对象所存放的地址是不一样的,这边的"equals"比较的就是值,这里为什么比较的是值呢,equals里的重写了equals的方法。附上源码:

    publicboolean equals(Object obj) {

    if(objinstanceof Integer) {

    returnvalue == ((Integer)obj).intValue();

     }

    return false;}

    补充说明:Java String 和 new String()的区别

      栈区存引用和基本类型,不能存对象,而堆区存对象。==是比较地址,equals()比较对象内容。

    (1)String str1 = "abcd"的实现过程:首先栈区创建str引用,然后在String池(独立于栈和堆而存在,存储不可变量)中寻找其指向的内容为"abcd"的对象,如果String池中没有,则创建一个,然后str指向String池中的对象,如果有,则直接将str1指向"abcd"";如果后来又定义了字符串变量 str2 = "abcd",则直接将str2引用指向String池中已经存在的“abcd”,不再重新创建对象;当str1进行了赋值(str1=“abc”),则str1将不再指向"abcd",而是重新指String池中的"abc",此时如果定义String str3 = "abc",进行str1 == str3操作,返回值为true,因为他们的值一样,地址一样,但是如果内容为"abc"的str1进行了字符串的+连接str1 = str1+"d";此时str1指向的是在堆中新建的内容为"abcd"的对象,即此时进行str1==str2,返回值false,因为地址不一样。

    (2) String str3 = new String("abcd")的实现过程直接在堆中创建对象。如果后来又有String str4 = new String("abcd"),str4不会指向之前的对象,而是重新创建一个对象并指向它,所以如果此时进行str3==str4返回值是false,因为两个对象的地址不一样,如果是str3.equals(str4),返回true,因为内容相同。

    相关文章

      网友评论

          本文标题:1.浅谈“==”和equal的区别及其引发的思考

          本文链接:https://www.haomeiwen.com/subject/goembctx.html