Java中String两种不同创建方式的区别及intern的用法

时间:2023-01-31 18:10:26

一,

Java有两种创建字符串的方式,

        String str1 = "abc";
        String str2 = new String("abc");

用双引号创建和用new来创建。这两种方式创建出来的String存储的位置上不同的。当使用双引号方式时,相当于显式的声明了字符串的值(字面值),所以是作为一个常量,存储在方法区的常量池中。使用new方式来创建String时,JVM会在堆上分配一块区域,存储一个String对象,值为“abc”。

二,

String的==和equals是不同的,==比较的是两个String在内存中的地址是否相同,equals比较的是两个String的值是否相同。所以,如果String是用双引号方式创建,则两个String都指向常量池中的同一个位置,这时==是成立的,equals也成立。如果String是用new方式创建的,两个String如果不是指向堆上的同一个String对象,则==不成立,而如果值相同,则equals成立。如:

        String str1 = "abc";
        String str2 = new String("abc");
        String str3 = "abc";
        String str4 = str2;
        System.out.println(str1 == str3);//true
        System.out.println(str1.equals(str3));//true
        System.out.println(str2 == str3);//false
        System.out.println(str2.equals(str3));//true
        System.out.println(str2 == str4);//true, str4和str2指向同一个对象
        System.out.println(str2.equals(str4));//true

三,

两种不同方式创建的字符串,在进行运算时结果是不同的。如果一个String是由两个常量运算得到的,相当于用双引号创建出来,会存入常量池,如:

        String s = "a" + "b";
        String t = "ab";
        //以上两种创建方式,完全等价

如果运算时,等号右边有一个“运算数”不为常量(即使该“运算数”是用双引号创建的常量),则得到的结果相当于new创建的一个新的String对象,如:

        String str1 = "abc";
        String str2 = "abcd";
        String str3 = "abc" + "d";
        String str4 = str1 + "d";
        
        System.out.println(str2 == str3);//true
        System.out.println(str2.equals(str3));//true
        System.out.println(str2 == str4);//false
        System.out.println(str2.equals(str4));//true

四,intern()方法的用法。intern方法会返回一个字符串对应的常量值。在执行intern方法时,JVM会检查常量池中是否存在和该字符串相同的常量值,如果有,则返回该常量值,若没有,则创建该常量值,并返回。即,intern返回的是值常量池中的String,不是堆上的String,相当于用双引号创建String。

        String str1 = "abc";
        String str2 = "abcd";
        String str3 = "abc" + "d";
        String str4 = (str1 + "d").intern();
        
        System.out.println(str2 == str3);//true
        System.out.println(str2.equals(str3));//true
        System.out.println(str2 == str4);//true
        System.out.println(str2.equals(str4));//true
     //注意和上例的区别