JVM_04 字符串常量池(详解intern( ))
一、String的基本特性
String:字符串,使用一对""引起来表示。
String sl = “hello”;//字面量的定义方式
String s2 = new String(“hello”)
String声明为final的,不可被继承
String实现了Serializable接口:表示字符串是支持序列化的。 实现了Comparable接口:表示String可以比较大小
String在jdk8及以前内部定义了final char[],value用于存储字符串数据。jdk9时改为byte[ ]
String:代表不可变的字符序列。简称:不可变性。
通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中
字符串常量池中是不会存储相同内容的字符串的
- 1.String的String Pool 是一个固定大小的Hashtable,默认值大小长度是1009。如果放进StringPool的String非常多,就会造成Hash冲突严重,从而导致链表会很长,而链表长了后直接会造成的影响就是当调用String. intern时性能会大幅下降。
- 2.使用 -XX:StringTableSize可设置StringTable的长度
- 3.在jdk6中StringTable是固定的,就是1009的长度,所以如果常量池中的字符串过多就会导致效率下降很快。StringTableSize设 置没有要求
- 4.在jdk7中,StringTable的长度默认值是60013
- 5.jdk8开始,1009是StringTable长度可设置的最小值
二、String的内存分配
- 1.直接使用双引号声明出来的String对象会直接存储在常量池中(比如: String info = “abc” )
- 2.如果不是用双引号声明的String对象,可以使用String提供的intern()方法
三、字符串拼接操作
常量与常量的拼接结果在常量池,原理是编译期优化
常量池中不会存在相同内容的常量。
只要其中有一个是变量,结果就在堆中
(只有有一个是变量,那么它会在堆中创建一个StringBuilder,调用append( )方法进行添加操作,调用toString( )方法转换为字符串【toString( )方法其实就是:new String( )】)
- 1.JDK1.6,将这个字符串对象尝试放入串池
①. 如果字符串常量池中有,则并不会放入。返回已有的串池中的对象的地址
②.如果没有,它会在常量池中创建一个对象放入串池中,并返回串池中的对象地址 - 2.JDK1.7,将这个字符串对象尝试放入串池
①. 如果字符串常量池中有,则并不会放入。返回已有的串池中的对象的地址
②. 如果没有,它不会创建一个对象,如果堆中已经这个字符串,那么会将堆中的引用地址赋给它 - 3.test3、test4重点掌握
四、拼接操作与append的效率对比
五、intern()的使用 (超级重要)
前言:
- (1). 如果不是用双引号声明的String对象,可以使用String提供的intern方法:intern方法会从字符串常量池中查询当前字符串是否存在,若不存在就会将当前字符串放入常量池中
- (2). 比如: String myInfo = new String(“I love u”).intern(); 也就是说,如果在任意字符串上调用String. intern方法,那么其返回结果所指向的那个类实例,必须和直接以常量形式出现的字符串实例完全相同。因此,下 列表达式的值必定是true:(“a” + “b” + “c”).intern()== “abc”;
- (3). 通俗点讲,Interned String就是确保字符串在内存里只有一份拷贝,这样可以节约内存空间,加快字符串操作任务的执行速度。注意,这个值会被存放在字符串内部池(String
Intern Pool)
①. new String(“ab”)会创建几个对象 很重要
(new String(“ab”)会创建几个对象?看字节码,就知道是两个)
②. new String(“a”) + new String(“b”)呢? 很重要
(在常量池中是没有ab存在的 [很关键] )
-
对象1:new StringBuilder()
-
对象2: new String(“a”)
-
对象3: 常量池中的"a"
-
对象4: new String(“b”)
-
对象5: 常量池中的"b"
-
深入剖析: StringBuilder的toString(): 对象6 :new String(“ab”)
-
注意:强调一下,toString()的调用,在字符串常量池中,没有生成"ab" 没有ldc指令
四、总结String的intern()的使用
- JDK1.6,将这个字符串对象尝试放入串池
①. 如果字符串常量池中有,则并不会放入。返回已有的串池中的对象的地址
②.如果没有,它会在常量池中创建一个对象放入串池中,并返回串池中的对象地址 - JDK1.7,将这个字符串对象尝试放入串池
①.如果字符串常量池中有,则并不会放入。返回已有的串池中的对象的地址
②.如果没有,它不会创建一个对象,如果堆中已经这个字符串,那么会将堆中的引用地址赋给它
intern题目:
六、intern()效率测试
大的网站平台,需要内存中存储大量的字符串。比如社交网站,很多人都存储:北京市、海淀区等信息。这时候如果字符串都调用 intern()方法,就会明显降低内存的大小。
/*** 使用intern()测试执行效率:空间使用上** 结论:对于程序中大量存在存在的字符串,尤其其中存在很多重复字符串时,使用intern()可以节省内存空间。**/ public class StringIntern2 {static final int MAX_COUNT = 1000 * 10000;static final String[] arr = new String[MAX_COUNT];public static void main(String[] args) {Integer[] data = new Integer[]{1,2,3,4,5,6,7,8,9,10};long start = System.currentTimeMillis();for (int i = 0; i < MAX_COUNT; i++) { // arr[i] = new String(String.valueOf(data[i % data.length]));arr[i] = new String(String.valueOf(data[i % data.length])).intern();}long end = System.currentTimeMillis();System.out.println("花费的时间为:" + (end - start));try {Thread.sleep(1000000);} catch (InterruptedException e) {e.printStackTrace();}System.gc();} }七、 StrtingTable的垃圾回收(了解)
/*** String的垃圾回收:* -Xms15m -Xmx15m -XX:+PrintStringTableStatistics -XX:+PrintGCDetails**/ public class StringGCTest {public static void main(String[] args) { // for (int j = 0; j < 100; j++) { // String.valueOf(j).intern(); // }//发生垃圾回收行为for (int j = 0; j < 100000; j++) {String.valueOf(j).intern();}} }总结
以上是生活随笔为你收集整理的JVM_04 字符串常量池(详解intern( ))的全部内容,希望文章能够帮你解决所遇到的问题。
- 上一篇: 中国剧本推理市场洞察2021
- 下一篇: 2022零售行业消费趋势新主张