String类底层的修改:
Jdk1.8时底层为char[],jdk1.9时,底层为byte[]。
String不可变性:
字符串常量池中是不会存储相同内容的字符串,是因为String 的String pool是一个固定大小的HashTable。
String类型常量池使用的两种方法;
String对应不同jdk版本的内存分配:
对StringTable进行调整的原因;
- PermSize默认比较小
- 永久代垃圾回收频率低
完全相同的字符串字面量应包含同样的unicode字符序列(包含同一份码点序列的常量),并且必须是指向同一个String类实例。
字符串拼接操作:
- 常量与常量的拼接,结果是在常量池,原理是编译期优化
- 常量池中是不允许放相同内容的常量的
- 拼接操作中,只要有一个是变量,结果就是在堆中。变量拼接的原理是StringBuilder
- 如果拼接的结果调用intern()方法,则主动将常量池中还没有的字符串对象放入池中,并返回此对象地址
此处的s是临时定义的,底层不一定叫s
使用字符串拼接方式和使用StringBuilder的append方法对比:
如果进行GC,需要花费额外的时间。
使用StringBuilder方式的改进空间:如果在实际开发过程中,如果基本确定前前后后添加的字符串长度不高于某个限定值highLevel的条件下,建议使用构造器:
StringBuilder s=new StringBuilder(highLevel);//new char[highLevel]
Intern()方法:
保证变量s指向的是字符串常量池中的数据的方法:
面试题:
1.new String(“ab”)会创建几个对象?
两个对象:
一个是在堆空间中通过new关键词创建的,另一个就是在字符串常量池中存放的“ab”对象(字节码指令:ldc)
2.new String(“a”)+new String(“b”)会创建几个对象?
对象1:new StringBuilder
对象2:new String(“a“)_
对象3:常量池中的“a“
对象4:new String(“b”)
对象5:常量池中的“b“
深入剖析:StringBuilder中的tostring()方法
对象6:new String(“ab”)
强调一下:tostring()的调用,在字符串常量池中,没有生成“ab“
代码分析:
Jdk7时:此时常量池中并没有创建“11“,而是创建了一个指向堆空间中new String(“11”)的地址。
题目扩展:
总结intern():
题目扩展:
图解:
Jdk6中:
Jdk7/jdk8中:
如果开始时常量池中有“ab“:
练习2:
对于程序中大量存在的字符串,尤其其中存在很多重复字符串时,使用intern()方法可以节省内存空间。
G1中的去重操作:
具体实现(了解):
开启去重操作的指令: