8. StringTable

it2023-01-12  51

一、String 的存储结构和StringPool

1.1 String存储结构变更

​ String在jdk9中底层不再使用char型数组,而是使用byte型数组,因为在实际开发中发现大多数字符串中存储的都是Lantin-1(每个字符一个字节),这些字符只需要1byte的空间就可以存储。

1.2 String的基本特性

字符串常量池中不会存储相同内容的字符串;String的String Pool是一个固定大小的Hashtable(里面只存储String的引用,并不存储实际内容,实际内容存储在运行时常量池),默认大小长度是60013。如果长度过短,就会因为String的数量增加从而导致链表变长,效率降低;使用-XX:StringTableSize可设置StringTable的长度,1009是可设置的最小长度。

二、String 的内存分配

​ 看 6. JVM 方法区的3.1

三、字符串拼接操作

常量与常量的拼接结果在常量池,原理是编译期优化;常量池中不会存在相同内容的常量;只要其中有一个是变量,结果就在堆中,变量拼接的原理是StringBuilder如果拼接的结果调用intern()方法,则主动将常量池中还没有的字符串对象放入池中,并返回此对象的地址。 public class Test3 { public static void main(String[] args) { String a = "abcdef"; String b = "abc"; String c = "def"; String d = "abc" + "def"; //"abc" + "def" 在编译期间就被优化成为了 "abcdef" String e = b + c; //本质上执行的代码如下:String e = (new StringBuilder()).append(b).append(c).toString(); System.out.println(d == a); //true,因为d和a指向常量池中同一个地址 System.out.println(e == a); //false,因为e指向推空间中一个StringBuilder.toString()所在的地址,而a指向常量池中的一个地址 System.out.println(e.intern() == a); //true,e.intern()返回的是字符串"abcdef"在常量池中的地址 } }

注意:只有字符串变量在拼接的时候调用的才是StringBuilder,如果是final修饰的变量,则是编译期优化

public void test2() { final String a = "abcdef"; final String b = "abc"; final String c = "def"; String d = b + c; //b + c在编译期间就被优化成为了 "abcdef" System.out.println(d == a); //true }

四、intern() 的使用

4.1 intern() 的基本原理

​ 如果不是用双引号声明的String对象,可以使用String提供的intern方法:inter方法会从字符串常量池中查询当前字符串是否存在,若存在,返回已有的串池中对象的地址;若不存在,会把对象的引用地址复制一份,放入串池,并返回串池中的引用地址。

​ intern 就是确保字符串在内存里只有一份拷贝,这样可以节约空间,加快字符串操作任务的速度。

4.2 new String(“ab”) 会创建几个对象,new String(“a”) + new String(“b”) 呢

new String(“ab”) 会创建2个对象

new String本身字符串常量池中的ab

new String(“a”) + new String(“b”) 会创建5个对象

StringBuildernew String(“a”)常量池中的anew String(“b”)常量池中的b

4.2 关于 intern 的题目

练习一

public void test4() { String s = new String("1"); //在堆空间中创建了String对象,在常量池中创建了“1” String s1 = s.intern(); String s2 = "1"; System.out.println(s == s1); //false,s指向堆空间中的String对象,s1指向常量池中的 “1” System.out.println(s1 == s2); //true,s2也指向常量池中的“1” }

练习二

public void test5() { String s3 = new String("1") + new String("1"); //此时字符串常量池中只有 “1”,而没有 “11” s3.intern(); //jdk7开始,常量池中并没有创建“11”,而是创建一个指向堆空间中 new String("11")的地址 String s4 = "11"; //s4变量记录的地址:指向堆空间中 new String("11")的地址 System.out.println(s3 == s4); //jdk6:false,jdk7/8:true }

练习三

public void test6() { String s5 = "11"; String s6 = new String("1") + new String("1"); String s7 = s6.intern(); System.out.println(s5 == s6); //false,因为“11”已经先创建了,所以调用s6.intern后,并不会在常量池中创建指向s5地址的项 System.out.println(s5 == s7); //true }
最新回复(0)