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相等。