(一)String中的intern方法的实现原理(jdk8中):
以下面代码为例
public class InternTest{
String s1 = new String("a") + new String("b");
s1.intern();
String s2 = "ab";
System.out.println(s1 == s2);//true
}
(1)String s1 = new String(“a”) + new String(“b”);
①创建StringBuilder对象,用于字符串拼接;
②在堆中创建了String(“a”)对象,然后在字符串常量池中放入字符串"a",再做StringBuilder的append操作;
③在堆中创建了String(“b”)对象,然后在字符串常量池中放入字符串"b",再做StringBuilder的append操作;
④对StringBuilder做toString操作,在堆中创建了一个String(“ab”)对象并把地址返回给s1,并没有在字符串常量池中放入"ab"。
(2)s1.intern();
常量池中不直接放入"ab",而是引用堆中已经创建的String(“ab”)。
(3)String s2 = “ab”;
s2指向字符串常量池中的"ab",实际上是指向了堆中的String(“ab”)。
(二)此外,如果程序中存在大量的字符串,并且很多字符串是重复的,那么使用intern()将可以节省空间,因为重复的数据都会从常量池中取得。
public class InternTest2{
static final int MAX_COUNT = 1000 * 1000;
static final String[] arr = new String[MAX_COUNT];
public static void main(String[] args){
Integer[] data = new Integer[]{
1,2,3,4,5};
for(int i = 0; i < MAX_COUNT; ++i){
//arr[i] = new String(String.valueOf(data[i % data.length]));//将会创建MAX_COUNT个String对象,并将地址存入arr数组中
arr[i] = new String(String.valueOf(data[i % data.length])).intern();//创建String对象和常量池中的字符串,然后把常量池中的字符串地址返回给到arr数组,然后刚刚创建的String对象可以被销毁
//5轮循环过后出现常量池中已存在的字符串,则可以把已存在的字符串的地址一一返回给arr数组,而不用像上一条那样,维护MAX_COUNT个String对象
}
}
}