废话不多说先看一段代码:
/**
* @author ZhuHao
* @date 2018/10/2 10:11
*/
public class TestString {
public static void main(String[] args) {
String str1 = "Hello";
String str2 = "Hello";
String str3 = new String("Hello");
String str4 = new String("Hello");
String str5 = new String("Hello").intern();
System.out.println(str1==str2);
System.out.println(str1==str3);
System.out.println(str3==str4);
System.out.println(str1==str5);
}
}
运行后控制台输出结果如下:
true
false
false
true
str1和str2用了直接赋值方法,实例化时只会开辟一块堆内存空间,并且会自动保存在字符串常量池中以供下次重复使用(实例化str2时会在常量池中查找"Hello",有则直接把引用复制给栈Stack中的str2,如果没有才会在Heap堆里重新创建一个对象,然后把引用驻留到字符串常量池,再把引用复制到栈Stack中);
str3和str4用了构造方法赋值,每次赋值会开辟两块堆内存空间,其中有一块空间将成为垃圾(直接使用new String("Hello")相当于先在堆内存创建一个匿名对象"Hello",然后在关键字new后在堆内存上新开辟一个"Hello",并将引用复制到栈Stack中的str3和str4,而原先堆内存中第一次创建的"Hello"就成为了垃圾,本例中字符串常量池中已经存在"Hello"),而且不会自动入字符串常量池;
str5重复了这一步骤,但是最后使用了intern()方法手工入池;
在开发中一般使用直接赋值法;
参考知乎“胖君”同学的回答,图也引自同链接: