Java中的string类型一直是一个热点问题,也是Java面试问题中的常客。string类型涉及到常量池,堆等方面,是考察Java基础的很好的问题。我整理了一些常见的情况,做了一个总结。
public void run1() {
String str1 = "abc";
String str2 = new String("abc");
System.out.println(str1 == str2);
}
这是最基本,也是最常见的问题了。创建str1时,JVM把字面值常量 "abc"加入到位于堆的字符串常量池中,然后返回这个常量的引用,str1就是这个引用。创建str2时,JVM在堆中开辟一小块空间,用来容纳 "abc",然后返回这一小块空间的首地址,str2就是这个地址。
public void run2() {
//intern的作用,返回字符串在常量池中的引用,如果不能再常量池中找到该字符串,则把该字符串加到
String str1 = "abc";
String str2 = new String("abc");
String str3 = new String("abc");
System.out.println(str2 == str3);
System.out.println(str2.intern() == str3.intern());
System.out.println(str2 == str3.intern());
System.out.println(str1 == str2.intern());
}
- str2,str3是两个单独new出来的对象,在堆中各自占据一定的空间,因此str2==str3为false。
- intern()的作用是返回string类型对象在字符串常量池中的引用,如果这个string类型不在字符串常量池中,则把这个字符串添加到字符串常量池中,然后在返回这个在常量池中的字符串的引用。因此,虽然str2,str3在堆中占据不同的内存空间,但是调用intern()方法后,两个字符串都被拷贝到字符串常量中,而对于相同的字符串对象,字符串常量池又只会保留一个,因此str2.intern() 返回的结果就和str3.intern()相同
- 按照上面一点,str3.intern()返回的是str3在字符串常量池中的引用。因此str2 == str3.intern()为false
- 根据第二点,str1是常量池中"abc"的引用,str2.intern()最终返回的也是"abc"在常量池中的引用,因此str1 == str2.intern()为true
public void run3() {
String str1 = "abc";
String str2 = new String("abc");
str2.intern();
System.out.println(str1 == str2);
}
网上对intern()的解释大多都是会把字符串从堆中添加到字符串常量池中。但是先执行intern(),然后再比较str1 == str2,结果却是false。因为str2仍旧是JVM在堆中给"abc"对象所分配的内存块的首地址。
public void run4() {
//两个字符串常量拼接,jvm直接把结果加入到常量池中。
//如果一个字符串常量引用拼接一个字符串常量,调用的实际上是StringBuilder.append,是创建一个新的字符串
String str1 = "abc";
String str4 = "ab";
String str5 = "ab" + "c";
String str6 = str4 + "c";
//final变量在编译后会直接替换成对应的值
final String str7 = str4;
String str8 = str7 + "c";
final String str9 = "ab";
String str10 = str9 + "c";
System.out.println(str1 == str5);
System.out.println(str1 == str6);
System.out.println(str1 == str6.intern());
System.out.println(str1 == str8);
System.out.println(str1 == str10);
}
- 首先是final关键字问题。被final关键字修饰的对象,在编译的时候会被替换成字面常量。因此str10=str9+"c" 就相当于str10="ab"+"c" ,结果为true
- 如果用字符串常量拼接成一个新的字符串,JVM会把最终的结果存储到字符串常量池中。而如果用一个字符串引用来参与拼接,则相当与调用stringBuilder.append方法,最终的对象会创建在堆中。
- 第三个比较再次说明intern()返回的是字符串在常量池中的引用。