java中的String解析

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."<init>":()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

 

posted @ 2020-09-25 15:58  兵哥无敌  阅读(300)  评论(0编辑  收藏  举报