-
首先确定一点无论是jdk1.6还是1.7
String c = "ab";
这种字面赋值都是直接在字符串常量池生成一个字符串对象的。
-
String a = new String("a"); String b = "a"; System.out.println(a == b);
输出的是
false
。
这说明在用new生成字符串对象的时候,只会在堆中生成字符串对象。而不会在字符串常量池生成对象,也没有引用。 -
明确:
1、在jdk1.6中,intern()方法是先查找字符串常量池是否含有当前字符串,如果没有,那么就在字符串常量池中创建 该字符串,并且返回该字符串在字符串常量池中的引用。
2、在jdk1.7中,intern()方法也是先查找字符串常量池中是否含有当前字符串,如果没有,将该字符串在堆的引用存在常量池中。并且返回该字符串在堆中的引用。
这是因为在jdk1.6中字符串常量池在永生代(方法区)中,而在jdk1.7字符串常量池转移到了堆中。
《深入理解Java虚拟机:JVM高级特性与最佳实践(第3版) (华章原创精品)》书中原话。
上述即可解释下面的例子:
需要注意的是,这里看作之前字符串常量池就有java这个字符串了。
-
看下面示例
String a = new String("ab"); String b = new String("ab"); String c = "ab"; String d = "a" + "b"; String e = "b"; String f = "a" + e; System.out.println(b.intern() == a); System.out.println(b.intern() == c); System.out.println(b.intern() == d); System.out.println(b.intern() == f); System.out.println(b.intern() == a.intern());
输出结果为
false true true false true
由运行结果可以看出来,b.intern() == a和b.intern() == c可知,采用new 创建的字符串对象不进入字符串池,并且通过b.intern() == d和b.intern() == f可知,字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象。如果无此对象,则先创建此对象,再让对象引用指向此对象。
5.我们看下面示例:这个类似于我们上面在书中看到的那个例子
输出是false,但是按照我们上述的推理来说他应该输出的true才对。
于是我查阅各种资料发现可能是因为字符串a在常量池中已经被创建过了。尝试下面写法:
String str1 = new String(new char[]{'a'});
System.out.println(str1.intern() == str1);
输出的是True。
灰色的应该就是字符串常量池已经有这个字符串了