1.0简介intern()

intern()是一个Native方法。

它的作用是:如果常量池中已经包含一个等于此String对象的字符串,则返回常量池中的这个字符串String对象;否则返回,将此String对象包含的字符串添加到常量池中,并且返回此String对象的引用。

源码:注意该方法有native修饰。

* @return a string that has the same contents as this string, but is
* guaranteed to be from a pool of unique strings.
*/
public native String intern();
1.1实验代码以及结果分析
public class Main {
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);
}
}

结果:

图片.png

产生该原因是:计算机软件这个字符串在jvm常量池中,本来是不存在的,在jdk1.7后,对于新增的字符串对象不会再复制字符对象到常量池中,而是把字符串对象的引用复制到常量池中,此时返回的是计算机软件字符串对象的引用。所以和str1的值相等。(注意:str1的值也是计算机软件字符串对象的引用);而java字符串会发生false的原因时,jvm启动时,常量池就有java字符串对象,所有返回的是在常量池的java字符串的引用。而str2的值是java字符串在堆上对象的引用。所以不相等。

至于为什么java字符串对象一开始就存在于常量池的原因:

参考:https://www.zhihu.com/question/32672669

各位大佬各有说法,我这里说的jvm启动就存在,估计有错。


1.2从这里引起对String和StringBuilder的思考

图片.png

其中result = new StringBuilder(result).append(str).toString();这句话的执行除了new出一个StringBuilder对象外,还有在调用toString后,会new 出一个String对象。但是并不会在常量池添加这个String对象,只有使用result.intern()方法后,才会添加。

1.2关于String的一些整理:

表达式

内存变化

String str1 = "常量";

这个在常量池中添加"常量",并且str1指向它在常量池的地址

String str2 = new String("winney");

"winney"对象在堆上',str2指向"winney"对象在堆上的地址

str2.intern();

"winney"会被放入常量池中,并且存的值和str2一样,是"winney"的地址值

1.3总结

图片.png

关于第一个false:str1指向的常量池上值为winney的String对象的引用,而str2指向的是java堆上的值为winney的String对象的引用;

关于第一个true:str1.intern()返回的字符串内容和str1相同的并且是在常量池的字符串,由第一行代码知,str1指向的地址就是在常量池的引用,所以两者相等;

关于第二个true:str2.intern()返回的是与str2内容同是winney的并且保证存在在常量池的字符串,所以和str1相等。