关于String.intern()和new StringBuilder("").append("").toString() “java”在执行StringBuilder.toString()之前出现

关于《深入理解Java虚拟机》第2章的问题

在书中有这么一个例子:

public class RuntimeConstantPoolOOM{
	public static void main(String[] args){
		String str1=new StringBuilder("计算机").append("软件").toString();
        System.out.println(str1.intern()==str1);
        String str2=new StringBuilder("ja").append("va").toString();
        System.out.println(str2.intern()==str2);
	}
}

作者指出由于JDK1.7开始逐步“去永久代”,所以如果JDK1.6会返回两个false,JDK1.7运行则会返回一个true一个false。

因为JDK1.6中,intern()方法会把首次遇到的字符串实例复制到永久代中,返回的也是永久代中这个字符串的实例的引用,而StringBulder创建的字符串实例在Java堆上,所以必然不是同一个引用,将返回false。

在JDK1.7中,intern()的实现不会在复制实例,只是在常量池中记录首次出现的实例引用,因此返回的是引用和由StringBuilder.toString()创建的那个字符串实例是同一个。

str2的比较返回false因为”java”这个字符串在执行StringBuilder.toString()之前已经出现过,字符串常量池中已经有它的引用了,不符合“首次出现”的原则,而“计算机软件”这个字符串是首次出现,因此返回true。

那么就有疑问了,这个“java”字符串在哪里出现过呢?显然并不是直接出现在这个类里面。
在这里插入图片描述对System,ctrl+鼠标左键,在System类中可以看到,此类是直接调用的
在这里插入图片描述在该类中,查看initializeSystemClass方法,其中有个Version
在这里插入图片描述查看你Version源码
在这里插入图片描述可以看到常量池在运行时,已经自动加入了“java”字符串,所以常量池中已经有了“java”字符串。

如若有错误,欢迎大牛们提出,感谢!!!

发布了24 篇原创文章 · 获赞 0 · 访问量 611

猜你喜欢

转载自blog.csdn.net/weixin_43896829/article/details/103840822
今日推荐