在 jdk1.6(含)之前也是方法区的一部分,并且其中存放的是字符串的实例;
在 jdk1.7(含)之后是在堆内存之中,存储的是字符串对象的引用,字符串实例是在堆中;
public static void main(String[] args) {
String s1=new String("a");
s1.intern();
String s2="a";
System.out.println(s1==s2);
String s3=new String("a")+new String("a");
s3.intern();
String s4="aa";
System.out.println(s3==s4);
}
上面这段代码,在1.6返回false,false,在1.8返回false,true。下面分析一下为什么
两个前提
- intern()方法是将字符串放到常量池中。
- java中只要发现双引号包围的字符串就会将之放到常量池中。
所以第一行代码String s1=new String("a");
这时候字符串a已经被放到常量池了。第二行是无效操作。而s3只有等到调用intern才被放到常量池,在JDK1.6中是直接复制进常量池,所以s3,s4不是同一个,1.8中常量池放的是堆中s3的引用,而s4又是引用常量池的,所以还是引用了堆,从而s3,s4是同一个。