一、作用:
- 当常量池中不存在"abc"这个字符串的引用,将这个对象的引用加入常量池,返回这个对象的引用。
- 当常量池中存在"abc"这个字符串的引用,返回这个对象的引用;
二、上代码
public class day01_6 {
public static void main(String[] args) {
String str1 = new StringBuilder("计算机").append("软件").toString();
String str2 = str1.intern();
String str3 = new StringBuilder("ja").append("va").toString();
String str4 = str3.intern();
System.out.println(str1==str2);
System.out.println(str3==str4);
}
}
jdk1.8的输出答案是true和false。
jdk1.6的输出是两个false。
三、分析
在jdk1.6中
intern方法会把首次遇到的字符串复制到方法区中,返回的也是方法区这个字符串的引用。
而由StringBuilder创建的字符串实例在Java堆上,所以必然不是一个引用,所以返回false
str1指向堆,str2指向方法区,所以返回结果返回false
同理,str3和str4的返回结果也为false
下来我们看一看jdk1.7的intern方法
jdk1.7的intern方法不会在复制实例,只是在常量池中记录首次出现的实例引用。
因此str2指向的引用其实就是str1指向Java堆中StringBuilder创建的字符串实例。所以返回结果为true
但是java这个字符串常量在编译期就已经在方法区的常量池中了,不符合首次出现,所以str4指向的是常量池中的java字面量
所以返回结果为false。
问题又来了,java这个字面量为什么在编译期就出现在了常量池。我们可以进入System类中。看看有什么东西。
进入System类之后,我们发现这里有一个Version.init方法
再次进去查看
哇,这么多常量,包括java,版本号,此版本号,都已经加载到常量池中,所以当我们调用str3.intern()方法时,java字面量已经存在,不符合首次出现,所以返回false,同理,我们也可以试一试这里的字面量,发现返回都是false。
String类的一个intern方法,涉及到了Java堆,java运行时常量池,涉及面很广泛,如果你不了解,是不是很吃亏。
面试问题:
(1)现在当有人问 String str = new String(“abc”);创建了几个对象,常量池有abc字段是1个,常量池没有"abc"字段则是2个。
(2)String str=“abc”;创建了几个对象(如果常量池里面已经有对象了就是0个。如果没有就是1个);
(3)new String(“abc”).intern();创建了几个对象(如果常量池里面已经有该字符串对象了就是1个,如果没有就是两个)