基本特性
- 字符串,使用一对""引起来表示
- 声明为final的,不可被继承
- 实现了Serializable接口:表示字符串是支持序列化的
- 实现了Comparable接口:表示String 可以比较大小
- 在jdk8及以前内部定义了final char[] value用于存储字符串数据,jdk9改成了byte[]
- 代表不可变的字符序列
- 字符串常量池中是不会存储相同内容的字符串的
- String的String Pool是一个固定大小的HashTable,默认值大小长度是1009。如果放进String pool的String非常多,就会造成Hash冲突严重,从而导致链表会很长,而链表长了后会直接造成的影响就是当调用String.intern时性能会大幅下降。
- 使用-XX:StringTableSize可设置StringTable的长度
- 在jdk6中 StringTable是固定的,就是1009的长度,
- 在jdk7中,StringTable的长度默认值是60013
- 在jdk8开始,1009是可设置的最小值
内存分配
- String类型的常量池比较特殊。他的主要使用方法有两种
- 直接使用双引号声明出来的String对象直接存储在常量池中
- 如果不是用双引号声明的String对象,可以使用String提供的intern()方法
- jdk6 及以前,字符串常量池放在永久代
- jdk7,字符串常量池的位置调整到了java堆中
- jdk8,字符串常量池在堆
- StringTable为什么要调整?
- 字符串拼接操作
- 常量与常量的拼接结果在常量池,原理是编译期优化
- 常量池中不会存在相同内容的常量
- 只要其中有一个是变量(不是final修饰),结果就在堆中。变量拼接的原理是StringBuilder
- new StringBuilder之后调用append方法,最后调用toString()方法,在字符串常量池中没有生成最后结果的字符串
- 如果拼接的结果调用intern()方法,则主动将常量池中还没有的字符串对象放入池中,并返回此对象地址
- 为什么Stringbuilder 比 字符串拼接的效率高
- Stringbuilder的append方式:自始至终只创建了一个Stringbuilder对象,而使用String的字符串拼接方式:创建过多个Stringbuilder和String的对象
- 使用String的字符串拼接方式:内存中由于创建了较多的Stringbuilder和String的对象,内存占用更大;如果需要gc时,需要花费的时间更多
- 如果使用Stringbuilder时,尽量在new时指定大小
- intern的使用
- 如何保证变量指向的是字符串常量池中的数据呢
- String s = "xiaoyi" 字面量方式
- new String("xiaoyi").intern()
- jdk6中,将这个字符串对象尝试放入串池
- 如果串池中有,则并不会放入。返回已有的串池中的对象的地址
- 如果没有,会把此对象复制一份,放入串池,并返回串池中的对象地址
- jdk7起,将这个字符串对象尝试放入串池
- 如果串池中有,则并不会放入。返回已有的串池中的对象的地址
- 如果没有,则会把对象的引用地址复制一份,放入串池,并返回串池中的引用地址
- String s1 = new String("1")+new String("2"); s1.intern();String s2 = "11"; // s1 == s2 true
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)