后记
写完这篇文章后,发现自己对字符串其实并不是很熟悉,具体表现在这几个问题
所以我又写了一篇文章来补充上面的问题。占坑 还未更新
1、前提知识
Java程序的运行内存中有三个区域
- 堆:存储对象
- 栈:用于存储变量或常量的引用变量
- 常量池:存储字面量
2、创建String的过程
字符串的创建有两种方式
- 用字面量创建
String s = "abc";
- new一个字符串对象
String s = new String("abc");
字面量声明
- 先去常量池中,看是否存在这个字面量,如果存在,就把字面量在常量池中的地址返回给引用变量
- 如果不存在,就在常量池中开辟空间存储常量值,再把地址返回给引用变量
new一个字符串对象
- 先去常量池中,看是否存在这个字面量,如果存在,就在堆中创建空间,再把堆地址返回给引用变量
- 如果不存在,就在常量池中开辟空间存储常量值,再在堆中创建空间,把堆地址返回给引用变量
PS:引用变量指向堆地址,堆地址中的值是字符串在常量池的地址(C语言的指针)
总结创建过程
常量池中只会存在一个相同值的字面量
- 先去常量池中,看是否存在对应的字符串常量值,如果不存在,就创建一个
- 如果用字面量创建,就返回常量池中的地址
- 如果new一个对象,就在堆中开辟空间,在返回堆中的地址
3、字符串的不可变性
4、字符串拼接
String s1 = "a";
String s2 = "b";
String s3 = "ab";
存在三种情况
String s5 = s1 + s2;
变量 + 变量String s6 = s1 + "b";
变量 + 字面量(常量)String s7 = "a" + "b";
字面量(常量)+ 字面量(常量)
验证
System.out.println(s5 == "ab"); false
System.out.println(s6 == "ab"); false
System.out.println(s7 == "ab"); true
System.out.println(s7 == s5.intern()); true
final String a = "aa";
String b = "bb" + a;
System.out.print(b == "aabb");true
总结
- 字面量与字面量拼接的字符串会放在常量池,并且常量池中不会存在相同值的字面量
- 只要有一个字符串变量参与拼接,结果存放在堆中
- 调用
intern()
时,如果在常量池中已经存在字符串字面量,就返回其在字符串常量池中的地址,如果没有就返回在其在堆中的地址,并在常量池中创建一份。