美文网首页
Java中==和equals的区别

Java中==和equals的区别

作者: 囧略囧 | 来源:发表于2018-09-27 10:36 被阅读0次

    1、双等号(==)

    双等号(==)的作用有两种:

    1)对于基本数据类型(byte, short, char, int, long, float, double, boolean),双等号(==)比较的是它们的值。

    2)对于符合数据类型,比如类、数组等,双等号(==)比较的是它们在内存中的地址。

    2、equals

    JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地 址,但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。

    也就是说复合数据类型在没有复写equals方法的时候,equals方法和“==”的作用是相同的。

    下面来看几个例子:

    public class Main {
        public static void main(String[] args) {
            String s1 = "abc";
            String s2 = "abc";
            System.out.println(s1 == s2);
        }
    }
    

    输出结果为

    true
    

    这不代表在Sting中==比较了双方的值。

    通过字面量赋值创建字符串时,会优先在常量池中查找是否已经存在相同的字符串,倘若已经存在,栈中的引用直接指向该字符串;倘若不存在,则在常量池中生成一个字符串,再将栈中的引用指向该字符串。(由于Java中String的不可变性,即使多个引用指向了相同的Sting也不会产生问题。)

    而通过new的方式创建字符串时,就直接在堆中生成一个字符串的对象。

    我们新建了s1,当新建s2时发现“abc”已存在,则s2与s1指向了常量池中的同一个字符串,地址相同,所以s1==s2返回了true。

    于是也很好理解下面的代码,使用new String在堆中新建一个String不同于常量池中的已有字符串,则s1和s2指向字符串的地址自然也不同。

    public class Main {
        public static void main(String[] args) {
            String s1 = "abc";
            String s2 = new String("abc");
            System.out.println(s1 == s2);
        }
    }
    

    输出结果为

    false
    

    当对代码做一个小改动后

    public class Main {
        public static void main(String[] args) {
            String s1 = "abc";
            String s2 = new String("abc").intern();
            System.out.println(s1 == s2);
        }
    }
    

    输出结果为

    true
    

    当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。

    s2调用intern方法,会将str2中值(“abc”)复制到常量池中,但是常量池中已经存在该字符串(即s1指向的字符串),所以直接返回该字符串的引用,因此s1==s2输出为true。

    下面我们来看一段有意思的代码

    public class Main {
        public static void main(String[] args) {
            String base = "ab";
            String s1 = "abc";
            String s2 = "ab" + "c";
            String s3 = base + "c";
            System.out.println(s1 == s2);
            System.out.println(s1 == s3);
        }
    }
    

    输出结果为

    true
    false
    

    通过反编译我们可以知道,

    s1==s2是从常量池中取出“abc”放到栈中,并将“abc”的引用值赋值给s1,从常量池中取出“abc”放到栈中,并将“abc”的引用值赋值给s2。s1==s2 肯定会返回true,因为s1和s2都指向常量池中的同一引用地址。所以其实在JAVA 1.6之后,常量字符串的“+”操作,编译阶段直接会合成为一个字符串。

    s1==s3时会先生成一个StringBuilder,加载第一个参数的值,即“base”,然后调用StringBuilder对象的append方法,之后加载常量池中的“c”到栈中,调用StringBuilder.append方法,最后调用StringBuilder.toString方法生成字符串并将其引用赋值给s3。因为s3实际上时StringBuilder.append()生成的结果,所以与s1不相等,结果返回false。

    但是如果在base前加final

    public class Main {
        public static void main(String[] args) {
            final String base = "ab";
            String s1 = "abc";
            String s3 = base + "c";
            System.out.println(s1 == s3);
        }
    }
    

    输出结果为

    true
    

    对于final字段,编译期直接进行了常量替换,而对于非final字段则是在运行期进行赋值处理的。

    因为s1和s3指向的都是常量池中的第三项,所以s1==s3返回true。

    相关文章

      网友评论

          本文标题:Java中==和equals的区别

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