Java中String两种不同创建方式的区别及intern的用法
一,
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 //注意和上例的区别