对intern()的理解
intern()的执行逻辑为:
在字符串常量池找和字符串对象的值相等的一个对象或引用 如果找不到,就创建,
如果堆里有找的对象,就创建一个引用,该引用指向堆里要找的对象,并返回该引用 如果堆里没有,就创建一个字符串对象并返回
先看这道题
String s = new StringBuilder("计算机").append("软件").toString();
Assert.assertTrue(s.intern() == "计算机软件");
Assert.assertTrue(s.intern() == s);
Assert.assertTrue(s == "计算机软件");
运行结果是成功的,2-4行全部为true。这里要注意到intern()的距离含义和逻辑
比如该题的执行逻辑
-
第一行在字符串常量池中创建了“计算机” 和 “软件” 这两个对象,最终将s 指向堆中的一个String对象,该对象的value 字段指向的堆中的 “计算机软件” 对象。
-
第二行 s .intern() 首先在字符串常量池中找 “计算机软件”,没找到,然后发现堆中有“计算机软件” 这个对象,于是在字符串常量池中创建了一个引用指向该对象,并将该引用返回; “计算机软件”同样在字符串常量池中找,找到了之前的指向“计算机软件” 的引用,也返回,所有 s .intern() == “计算机软件” 结果为true。
-
同样s也指向同一个"计算机软件"对象,所以都是true。
要证实这点,还可以再看以下代码
String s = new StringBuilder("计算机").append("软件").toString();
// 先执行"计算机软件",会字符串常量池生成一个字符串对象,之后intern返回是字符串对象的引用
Assert.assertTrue("计算机软件" == s.intern());//true
Assert.assertTrue(s.intern() == s);//false
Assert.assertTrue(s == "计算机软件");
当先执行"计算机软件"时,字符串常量池中就创建了"计算机软件"对象,这时intern()返回的就是字符串常量池中的"计算机软件"对象,和s指向的字符串就不是同一个了。
同样成立以下代码
String s = new StringBuilder("计算机软件").toString();
Assert.assertTrue(s.intern() == "计算机软件");//true
Assert.assertTrue(s.intern() == s);//false