可能有错误理解
String s1 = new String(“abc”);这句话创建了几个字符串对象?
以学习自java guide为前提 进行 推理
前提知识点:
字符串常量池:保存的是字符串对象的引用,该引用会指向字符串对象(比如字符串对象“abc”,而不是String对象,这个应该是区分开的)。
问题:String s1 = new String(“abc”);这句话创建了几个字符串对象?
我看的所有博客,都是说常量池中存了字符串实例,然后我就。。。。=_=
原文:
1、如果字符串常量池中不存在字符串对象“abc”的引用,那么会在堆中创建 2 个字符串对象“abc”。
堆中创建String对象,堆中创建字符串“abc”对象。
2、如果字符串常量池中已存在字符串对象“abc”的引用,则只会在堆中创建 1 个字符串对象“abc”。
我的逻辑:按照java guide说的为基础
按照1的说法,关键对象有:字符串对象的引用(在常量池中),String对象(new出来的),字符
串“abc”对象,在字符串数据中,要这三者齐全才是正常的。
串起来<指针指向>就是:栈中的引用s指向堆里的String对象,String对象指向常量池的字符串对象引用,字符串
对象引用最后指向堆里的字符串对象“abc”;
那么,在2中,就没有了String对象。难道不是常量池里有字符串对象“abc”的引用,它已经指向了“abc”,所以我认为应该是创建了String对象,然后去指向这个常量池的引用。这是和java guide 不同的地方
理解成,返回常量池中的对应字符串对象引用就可以了
如果引用不存在,就要创建;如果存在,直接返回。
等价于:String s = "abc"
String s = "abc"
, 这里是,s直接指向了常量池中字符串对象“abc”的引用。
总结如注释所示:
System.out.println("字符串常量相加");
// 对于编译期可以确定值的字符串,也就是常量字符串 ,jvm 会将其存入字符串常量池。
String str1 = "str";
String str2 = "ing";
// 字符串相加 就是StringBuilder的append方法,再toString方法返回一个新的String对象
// 字符串常量 拼接得到的 字符串常量 在 编译阶段 就已经被存放字符串常量池
// 在程序编译期就可以确定值的 常量,会直接使用字面值(相加后的字符串常量)替换,这叫常量折叠。而引用不能确定,所以不可以被这样优化。
String str3 = "str" + "ing"; // 可以常量折叠,等价于 String str3 = "string"
// 引用的值在程序编译期是无法确定的,编译器无法对其进行优化。
String str4 = str1 + str2; //当使用final修饰str1和str2时,那str1和str2就是常量了,就可以确定值,然后被优化
String str5 = "string";
System.out.println(str3 == str4);//false
System.out.println(str3 == str5);//true
System.out.println(str4 == str5);//false