看过很多文章介绍,关于什么样的字符串会存放在字符串常量池中:
1、单独使用””引号创建的字符串都是常量,编译期就已经确定存储到String Pool中。
2、使用只包含常量的字符串连接符如”aa”+”bb”创建的也是常量,编译期就能确定已经存储到StringPool中。
3、运行期调用String的intern()方法可以向String Pool中动态添加对象。
关于前两点其实有些问题。我们可以通过一些代码来看一下:
(1)首先创建一个TestStringPool类:
public class TestStringPool {
public TestStringPool() {
System.out.println("调用构造器");
}
private void test(){
String s = "11";
}
}
(2)再创建一个TestStringPool2 类
public class TestStringPool2 {
public static void main(String[] args) {
TestStringPool pool =new TestStringPool();
String s1 =new String("1") +new String("1");
String s2 = s1.intern();
String s3 ="11";
System.out.println(s1 == s3);
System.out.println(s1 == s2);
System.out.println(s2 == s3);
}
}
运行TestStringPool2的主方法。此时TestStringPool 类已经被加载进虚拟机了。该类中有一个字面量是“11”字符串。按照文章开头第1点,则此时字符串常量池中应该已经有“11”这个字符串了。
但上述代码在JDK1.7的环境中运行结果为:
调用构造器
true
true
true
也就是说,在代码运行到String s2 = s1.intern()时,发现字符串常量池中没有“11”这个字符串,于是把s1的的引用放入了字符创常量池中的StringTable中,并将改引用地址返回给s2.
我们在来改一下代码:把Strings3 ="11";向上移到String s1 =new String("1") +new String("1")之前变成一下代码:
TestStringPoolpool =new TestStringPool();
String s3 ="11";
String s1 =new String("1") +new String("1");
String s2 = s1.intern();
System.out.println(s1 == s3);
System.out.println(s1 == s2);
System.out.println(s2 == s3);
这时候的运行结果是:
调用构造器
false
false
true;
因为在String s2 = s1.intern()时,发现“11”在字符串常量池中有了,就不会将s1的引用放入StringTable中。
所以,这个字符串在什么时候会放入字符创常量池呢?应该是在使用到它的时候。并不是说编译时期就会放入。当然,如果你在TestStringPool类中加个成员变量String s = “11”或者 String s= new String("11")的话,其在new的时候,构造器就会去运行这个字符串的创建,并将其放入字符串常量池中。
ps:注意JDK1.6跟1.7的inter方法是有区别的,1.6的时候,字符串常量池还在方法区中,1.7则是在堆中的。