1、字符串常量池与intern方法
字符串常量池
jdk6中字符串常量池在永久代,从jdk7开始放到了堆空间中(是堆中又划分了一块区域,注意这个细节!)。
字符串对象创建方式
1、直接使用双引号声明出来的String对象
会去字符串常量池中找(通过equals)是否已存在该对象或者它的引用(保存了引用的情况),不存在直接创建在常量池中,存在就指向存在的。
String s1 = "abc";//去字符串常量池中找“abc”,没有找到然后在字符串常量池创建了一个字符串对象
String s2 = "abc";//找到了直接指向字符串常量池的对象
s1 == s2 //true
2、通过new String()创建
会在堆的字符串常量池和常量池之外各创建一个对象
3、+号拼接
+号拼接两边只要有一个变量最后拼接出来的字符串对象都是只在堆中常量池之外,常量池中没有。
注意:拼接的怎样才算变量和常量?对象或基本类型变量都算变量,final修饰基本类型就为常量,修饰引用变量该引用变量拼接还是算变量拼接
//两个变量一共创建了6个对象,多的一个就是StringBuilder。(常量池没有)
String s = new String("ab") + new String("cd");
//一个变量都没有:两个都是常量(字符串本身是常量),此时创建的对象在字符串常量池中
final int a = 1;
String s2 = a + "cd"
String s = "ab" + "cd";
4、对象的toString方法
这个得看toString的具体实现,是用的new String()还是字符串与变量拼接。
intern方法
intern方法的作用是如果字符串常量池已经包含一个等于(通过equals方法比较)此String对象的字符串,则返回字符串常量池中这个字符串的引用, 否则将当前String对象的引用地址(堆中对象的引用地址)添加(或者叫复制)到字符串常量池中并返回
(JDK7之前字符串常量池不在堆中,因此那时候不是存引用,而是新创建一个相同的对象在方法区的字符串常量池中)
代码讲解
加号拼接、引号创建两者与intern混合使用
int n = 2;//变量
String s1 = n + "b";//只要有一个变量最后拼接出来的字符串对象都是只在堆中常量池之外,常量池中没有。
String s2 = s1.intern();//intern检查到常量池没有该字符串于是就将该字符串的地址引用保存到常量池中并返回
System.out.println(s1 == s2);//true,两个引用变量实际指向的都是在堆中字符串常量池外的实际字符串变量地址
String s = "2b";//通过引号创建会先去常量池找是否存在,发现存在一个引用与“2b”进行equals相比是相等的,于是返回该引用
System.out.println(s == s1);//true,s就是常量池中“2b”的引用,即s2,而常量池中这个引用由指向s1
//总结:仅在堆中字符串常量池之外创建了一个“2b”字符串对象,字符串常量池存的是它引用,引用变量s也是它的引用。
//我们去掉String s2 = s1.intern();这个语句又如何呢?
int n = 2;//变量
String s1 = n + "b";//只要有一个变量最后拼接出来的字符串对象都是只在堆中常量池之外,常量池中没有。
//String s2 = s1.intern();//没有执行intern方法说明字符串常量池中没有该对象也没有该对象的引用
String s = "2b";//通过引号创建会先去常量池找是否存在,发现不存在,于是在常量池中创建该字符串对象并返回引用
System.out.println(s == s1);//false,s就是常量池中创建的“2b”的引用,而s1中这个引用指向的是常量池之外的“2b”
隐形引号创建
String s = new String("a") + new String("b");//new String("ab")
String s2 = s.intern();//jdk6中,在串池中创建一个字符串“ab”
//jdk7/8中,串池中没有创建字符串“ab”,而是创建一个引用,指向new String("ab")
System.out.println(s2=="ab");//jdk6:true jdk8:true。“ab”表示用引号创建字符串“ab”,去常量池中找发现存在
System.out.println(s=="ab");//jdk6:false jdk8:true
最后总结
记住三点:
- +号拼接两边只要有一个变量最后拼接出来的字符串对象都是只在堆中常量池之外,常量池中没有。
- new String()会在堆的字符串常量池之外和池内都创建一个对象
- 直接引号创建会只在常量池中创建,而且创建之前会检查常量池中是否存在该内容的字符串对象或引用,存在就返回引用。