String s=new String("xyz")及String s="a"+"b"+"c"+"d";创建几个String对象的问题

时间:2023-02-16 11:34:16

首先让我们了解几个概念:

栈:由JVM分配区域,用于保存线程执行的动作和数据引用。

堆:由JVM分配的,用于存储对象等数据的区域。

常量池constant pool :在堆中分配出来的一块存储区域,用于存储显式 的String,float或者integer.这是一个特殊的共享区域,可以在内存*享的不经常改变的东西,都可以放在这里。

进入正题:

String a = "abc";①
String b = "abc";②

使用String a = "abc";的方式,可以在一定程度上提高程序的运行速度,因为JVM会自动根据栈中数据的实际情况来决定是否有必要创建新对象。
①代码执行后在Constant Pool中创建了一个值为abc的String对象,②执行时,因为Constant Pool中存在"abc"所以就不在创建新的String对象了。

String   c   =   new   String("xyz");①
String   d   =   new   String("xyz");②

让我们来看看这两句代码在内存中发生了什么,①Class被CLassLoader加载时,你的"xyz"被作为常量读入,在constant   pool里创建了一个共享的"xyz",然后当调用到new   String("xyz")的时候,会在heap里创建这个new   String("xyz");②由于constant   pool中存在"xyz"所以不再创建"xyz",然后创建新的new   String("xyz")。
对于String c = new String("xyz");的代码,与String a = "abc"不同的是一概在堆中创建新对象,不管其字符串值是否相等,是否有必要创建新对象,从而加重了程序的负担。 
程序1
String   s1   =   new   String("xyz");     //创建二个对象,一个引用 
String   s2   =   new   String("xyz");     //创建一个对象,并且以后每执行一次创建一个对象,一个引用 
程序2 
String   s3   =   "xyz";     //创建一个对象,一个引用   
String   s4   =   "xyz";     //不创建对象,只是创建一个新的引用

重要的是理解constant pool与new关键字

当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。(无论怎样都返回池中的对象)

下面的这个例子能帮助我们更深入的理解String的存储和赋值原理

String str1 = new String("123");
String str2 = "123";

String str3 = str1.intern();
        
       System.out.println((str1 == str2) +","+ (str3 == str2));

       输出 false,true
                
       String str4 = new String("234");
       String str5 = new String("234");
        
       String str6 = str4.intern();
       String str7 = str5.intern();
        
       System.out.println((str4 == str5) +","+ (str6 == str7));

       输出 false,true

对如下代码:

String s1 = "a";
String s2 = s1 + "b";
String s3 = "a" + "b";
System.out.println(s2 == "ab");
System.out.println(s3 == "ab");
第一条语句打印的结果为false,第二条语句打印的结果为true,这说明javac编译可以对字符串常量直接相加的表达式进行优化,不必要等到运行期去进行加法运算处理,而是在编译时去掉其中的加号,直接将其编译成一个这些常量相连的结果。
题目中的第一行代码被编译器在编译时优化后,相当于直接定义了一个”abcd”的字符串,所以,上面的代码应该只创建了一个String对象。写如下两行代码,
String s ="a" + "b" + "c" + "d";
System.out.println(s== "abcd");
最终打印的结果应该为true。

则对于String s="a"+"b"+"c"+"d";创建了1个对象而已;