JVM学习之StringTable

String类底层的修改:

Jdk1.8时底层为char[],jdk1.9时,底层为byte[]。

String不可变性:

字符串常量池中是不会存储相同内容的字符串,是因为String 的String pool是一个固定大小的HashTable。

String类型常量池使用的两种方法;

String对应不同jdk版本的内存分配:

对StringTable进行调整的原因;

  • PermSize默认比较小
  • 永久代垃圾回收频率低

完全相同的字符串字面量应包含同样的unicode字符序列(包含同一份码点序列的常量),并且必须是指向同一个String类实例。

字符串拼接操作:

  • 常量与常量的拼接,结果是在常量池,原理是编译期优化
  • 常量池中是不允许放相同内容的常量的
  • 拼接操作中,只要有一个是变量,结果就是在堆中。变量拼接的原理是StringBuilder
  • 如果拼接的结果调用intern()方法,则主动将常量池中还没有的字符串对象放入池中,并返回此对象地址

此处的s是临时定义的,底层不一定叫s

使用字符串拼接方式和使用StringBuilderappend方法对比:

如果进行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中的去重操作:

具体实现(了解):

开启去重操作的指令:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值