intern()的理解1:
由于jdk7之前,字符串常量池是在永久代里面的(即方法区),新创建出来的两个对象一个在堆中,一个在永久代中
intern()在jdk7之后,字符串常量池放在了堆里面,此时如果如果new String().intern(),由于堆中保存两个相同的对象很浪费,所以调用了intern()之后,只有一个对象实体,并且在字符串常量池中。
intern()的理解2:
如果在调用intern()之前,字符串常量池和堆区已经生成了对象,则再次调用intern()也不会消除原本堆区的对象。
答案:2个
- 第一个是由于new String(),在堆区创建了一个对象,这个对象是字符串常量池中的对象的地址
- 同时在字符串常量池中也创建了一个对象,这个对象就是ab实体
说的更加具体一点:
由于是new String(),首先会在堆区创建一个ab的对象,然后就会看字符串常量池中是否有这个对象,如果有的话,就不在字符串常量池中新创建对象,直接将这个堆区的对象指向字符串常量池中的对象即可;如果没有的话,就创建这个ab的对象,并且将这个新创建出来的对象的地址,返回给堆区这个对象。
拓展答案:
注意:
针对于new String()的一个特例!!!
这里通过看字节码文件可以看到,如果是StringBuilder中也调用了new String(),此时并没有这个StringBuilder中的 new String(),并没有在字符串常量池中创建新的对象(特例,记住即可)
面试题:
第一部分:
jdk 6 为false的原因:
s为堆空间的变量,由intern()了之后,1加载进了字符串常量池,但是s指向的是堆空间的对象,而s2此时指向的是字符串常量池中的对象,所以不一样;
jdk 7/8 为false的原因:
理由同上
第二部分:
String s3 = new String(1)+new String(1);
此时并没有在字符串常量池中创建对象,仅仅在堆区创建了一个对象,通过intern()之后,将11也加到了字符串常量池中
不同的是:
jdk 6:
由于此时的字符串常量池没有在堆中,而s3指的是堆区的对象,s4是字符串常量池中的对象,是false;
jdk 7/8:
由于intern()的特点,导致此时的堆中只会存在一个对象,所以s3和s4是同一个对象。
总结: