关于String s = new String("xyz")创建几个对象

        你知道在java中除了8中基本类型外,其他的都是类对象以及其引用。所以 "xyz"在java中它是一个String对象对于string类对象来说他的对象值是不能修改的,也就是具有不变性。

    先看: 

<span style="font-family:Courier New;">     String s = "Hello"; 
     s = "Java "; 
     String s1 = "Hello"; 
     String s2 = new String( "Hello"); </span>

上面s所引用的String对象怎么被修改过? 不是说具有不变形吗? 这是怎么回事?


别着急,往下看:


在jvm的工作过程中,会创建一片的内存空间专门存入string对象。我们把这片内存空间叫做 string池


String s = "Hello ";当jvm看到"Hello",在string池 创建string对象存储它,并将他的引用返回给s。 
s= "Java ",当jvm看到 "Java",在string池创建新的string对象存储它,再把新建的string对象的引用返回给s。而原先的 "Hello"仍然在string池内。没有消失,他是不能被修改的。

所以我们仅仅是改变了s的引用,而没有改变他所引用的对象,因为string对象的值是不能被修改的。


String   s1= "Hello";jvm首先在string池内里面看找不找到字符串 "Hello",找到,返回他的引用给s1,否则,创建新的string对象,放到string池里。这里由于s= "Hello"了,对象已经被引用,所以依据规则s和s1都是引用同一个对象。所以  s==s1将返回true。(==,对于非基本类型,是比较两引用是否引用内存中的同一个对象)


String   s2=String( "Hello");jvm首先在string池内里面看找不找到字符串 "Hello",找到,不做任何事情,否则,创建新的string对象,放到string池里面。由于遇到了new,还会在内存上(不是string池里面)创建string对象存储"Hello",并将内存上的(不是string池内的)string对象返回给s2。所以s==s2将返回false,不是引用同一个对象。

好,现在我们看题目: 
String    new   String( "xyz"); 
首先在string池内找,找到?不创建string对象,否则创建,  这样就一个string对象 
遇到new运算符号了,在内存上创建string对象,并将其返回给s,又一个对象

所以一共就有两个对象。


这里为大家举一个例子:

<pre name="code" class="html"><pre name="code" class="javascript"><span style="font-family:Courier New;">public class Test { </span>
       <span style="font-family:Courier New;">public static void main(String []args) { 
            String s1=newString("test");   //创建2个对象,一个Class和一个堆里面 
            String s2="test";              //创建1个对象,s2指向pool里面的"test"对象 
            String s3="test";              //创建0个对象,指向s2指想pool里面的那个对象 
            String s4=s2;                  //创建0个对象,指向s2,s3指想pool里面的那个对象 
            String s5=newString("test");   //创建1个对象在堆里面,注意,与s1没关系 

            System.out.println(s2=="test");//trues2=="test"很明显true 
            System.out.println(s2==s3);    //true,因为指向的都是pool里面的那个"test" 
            System.out.println(s2==s4);    //true,同上,那么s3和s4...:) 
            System.out.println(s1==s5);    //false,很明显,false 
            System.out.println(s1==s2);    //false,指向的对象不一样,下面详解 1 
            System.out.println(s1=="test");//false,难道s1!="tset"?下面详解 2 

            System.out.println("---------------"); 
            s1=s2; 
            System.out.println(s1=="test");//true,下面详解 3
     }
}</span>

 
 

说明:

1. System.out.println(s1==s2);

   s2指向的对象"test"是在string池( pool )里面,而s1指向的是堆里面的"test"对象(s1指向的内存区),所以返回  false. 

2. System.out.println(s1=="test");

   s1指向的是堆里面的"test"对象(s1指向的内存区),而"test"是程序刚刚建立的(其实是共用string池( pool )里面的那个已经创建了的"test"对象,也就是我们s2="test"时候,在string池( pool )里面创建的),所以s1指向的堆里的"test"对象和"test"(pool里面)并不是一样个对象,所以返回 false. 

3. System.out.print(s1 == "test");

   当我们s1=s2;的时候,很明显,把s2的指给了s1,s1指向string池( pool )里面的"test",这个时候,s2也指向了string池( pool )里面的"test"对象了,当System.out.println(s1=="test");时候,java虚拟机创建"test"对象,注意,其实没创建,和前面讲的一样,公用s1="test"创建的"test"对象(string池( pool )里面的),所以,s1=="test"(pool里面的),同样,s1=s2=s3=s4! 


而为什么在网上都说String s=new String("test");创建了2个对象?那可能因为它就写这么一句代码,误让人默认的认为执行代码之前并不实例任何一个String对象过(也许很多人不会这么想,),跟着别人或者不经思考的就说2个,斟是说存放在栈内存中专门存放String对象引用的s变量是一个对象!实在不可原谅! 

不管大家是怎么理解的,也不知道我的理解是否正确,必竟我也写出了我想法了,罗里罗嗦的说了一下,好了,大家明白就好!



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值