结论
先看结果图:
有可能新创建了一个,也有可能新创建了两个。
举例
String s1 = "123";
String s2 = "123";
System.out.println(s1 == s2);
以上程序的执行结果为:true,说明变量 s1 和变量 s2 指向的是同一个地址,都是指向的字符串常量池的地址
String s3 = new String("123");
String s1 = "123";
System.out.println(s1==s3);
以上输出为false,这里指向false并非是因为他们指向的不是同一个字符串,其实指向的也都是字符串常量池中的同一个字符串,但由于是new String(“123”),会在堆中创建一个对象s3,由s3指向字符串常量池中对象,==对比的是堆上的内存地址
String s3 = new String("123");
String s4 = new String("123");;
System.out.println(s3==s4);
以上输出为false,这里同上,原因是在堆中创建了两个对象s3和s4,==对比堆上的内存地址,因此为false
经典问题:new String(“123”)创建了几个对象?
有可能创建1个,也有可能创建2个
- new String 的方式会首先去判断字符串常量池,如果没有就会新建字符串,那么此时就在字符串常量池中创建了一个对象,并在堆中创建了它的引用指向字符串常量池,此时就是创建了2个对象。
- 如果已经存在,那么就只会在堆中创建一个对象指向字符串常量池中的字符串。
问:以下应该是true还是false呢
String str1 = new String("123");
String str2 = "1" + "23";//编译器优化,其实是"123"
String str3 = "23";
String str4 = "1" + str3;//str3是变量,编译器无法进行优化,使用StringBuilder拼接
System.out.println(str1 == str2); //false
System.out.println(str2 == str4); //false
以下是程序反编译结果:详情请看 常量池