String是java中的引用类型,底层维护了一个private final char[],具有不可变性。下面用实例帮助我们理解String
String s1 = "ab"
此时会在字符串常量池中创建空间存放"ab" ,这个常量池位于jvm中的哪一块区域?
String s2 = new String ("ab")
此时会先查看常量池中有没有"ab",如果有,则在堆中创建空间存储"ab",栈中的s2指向堆中的"ab";如果没有,则在常量池中存放“ab”,然后在堆中存放“ab”,栈中的s2指向堆中的“ab”
System.println.out(s1==s2) //false
String s3 = new String("ab")
同上,在堆中创建空间存储“ab”,s3指向“ab”
System.println.out(s2==s3) //false
String s4 = "ab"+"cd"
“ab”,“cd” 在编译期确定是字符串常量,通过查看编译后的字节码文件,可以看到s4 = "abcd"
源码:
public static void main(String[] args) {
String s1 = "ab"+"cd";
System.out.println("abcd" == s1);
}
D:\IDE_CODE\Yuu\target\classes\com\xiaoyu\work\webservice\axis>javap -c StringTest.class
Compiled from "StringTest.java"
public class com.xiaoyu.work.webservice.axis.StringTest {
public com.xiaoyu.work.webservice.axis.StringTest();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."":()V
4: return
public static void main(java.lang.String[]);
Code:
0: ldc #2 // String abcd
2: astore_1
3: getstatic #3 // Field java/lang/System.out:Ljava/io/PrintStream;
6: ldc #2 // String abcd
8: aload_1
9: if_acmpne 16
12: iconst_1
13: goto 17
16: iconst_0
17: invokevirtual #4 // Method java/io/PrintStream.println:(Z)V
20: return
}
System.println.out(s4=="abcd") //true
String s5 = "ab";
String s6 = "cd";
String s7 = s5+s6;
此时s5+s6在编译期不能确定内容,只有在运行期才能确定内容,因此会在堆中创建空间1存放“ab”,创建空间2存放“cd”,创建空间3存放“abcd”。s7指向“abcd”
+号拼接原理:先调用String.valueOf()->StringBuilder ->append()->toString 返回
System.println.out(s7=="abcd") //false
String s8 = new String("abcd")
System.println.out(s8==s7) //false