`
xijunhu
  • 浏览: 155835 次
  • 性别: Icon_minigender_1
  • 来自: 无锡
社区版块
存档分类
最新评论

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

阅读更多
摘自csdn:http://topic.csdn.net/u/20070828/10/43260254-04f1-4ac0-9da1-b48af45c2a83.html

总结的正确回答:

回复于:2007-08-29 17:13:09
哈哈,要理解这个,就要知道string类的工作原理。

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


看:
String   s= "Hello ";
s= "Java ";
String   s1= "Hello ";
String   s2=new   String( "Hello ");

啊,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   s   =   new   String( "xyz ");
首先在string池内找,找到?不创建string对象,否则创建,   这样就一个string对象
遇到new运算符号了,在内存上创建string对象,并将其返回给s,又一个对象

所以总共是2个对象



回复于:2008-01-08 16:05:10以上朋友回答有道理,但我也有自己的想法(与Class类联系,因为创建类的时候,Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的。注意,所有String类型的都共享该 Class 对象!),
String s1="test";//创建一个对象??
String s2=new String("test");//创建2个对象??

都不一定,相对而言:
大家应该知道,String s="test";是在constant pool里面创建对象的;而String s=new String("test");是在内存堆里面创建的!

如:
1:
public class Tets{
public static void main(String [] args)
{
String s1=new String("test");//创建2个对象

/*如果你的程序只有这么一段程序的话,那么首先先由Java 虚拟机及通过调用类加载器中的 defineClass 方法自动构造Class对象,因为在String s1=new String("test");之前并没有创建任何String类型的对象,所以先是Java 虚拟机及通过调用类加载器中的 defineClass 方法自动构造Class对象!这个是第一个对象,还有new String("test")在堆里面创建的一个,一共2个,假如,(String s1=new String("test");)它前面有代码并且创建过String类的实例,那么可以说:String s1=new String("test");这句代码,它只创建一个在对象,就是new出来的这个对象*/
}  
}


2:public class Tets{
public static void main(String [] args)
{
String s1="test";//创建2个对象
/*
同理,只有它一条代码,需要创建Class对象,并且把"test"对象放在pool中,一共产生 2个对象,一个是Class和在pool中的"test"对象,假如它前面的有代码并且创建过String类的实例,道理同上,没必要再重复,
*/
}  
}
一个例子:
public class Test
{
public static void main(String [] args)
{
String s1=new String("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=new String("test");//创建1个对象在堆里面,注意,与s1没关系

System.out.println(s2=="test");//true s2=="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,指向的对象不一样,下面再说
System.out.println(s1=="test");//false,难道s1!="tset"?下面再说

System.out.println("---------------");

s1=s2;
System.out.println(s1=="test");//true,下面说
}
}
说明:1,System.out.println(s1==s2);很明显,s2指向的对象"test"是在pool里面,而s1指向的是堆里面的"test"对象(s1指向的内存区),所以返回false.
2,System.out.println(s1=="test");s1指向的是堆里面的"test"对象(s1指向的内存区),而"test"是程序刚刚建立的(其实是共用pool里面的那个已经创建了的"test"对象,也就是我们s2="test"时候,在pool里面创建的),所以s1指向的堆里的"test"对象
和"test"(pool里面)并不是一样个对象,所以返回false.
3,当我们s1=s2;的时候,很明显,把s2的指给了s1,s1指向pool里面的"test",这个时候,s2也指向了pool里面的"test"对象了,当System.out.println(s1=="test");时候,java虚拟机创建"test"对象,注意,其实没创建,和前面讲的一样,公用s1="test"创建的"test"对象(pool里面的),所以,s1=="test"(pool里面的),同样,s1=s2=s3=s4!

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

是于我的理解是否正确,也并不重要了,必竟我也写出了我想法了,再说了,我只是3流大学,不,是N流学校的学生罢了,再说了,我又不是SUN公司的头头,连个职工都不粘边,靠!
罗里罗嗦的说了一下,好了,大家应该明白我的意思了.


回复于:2008-03-16 23:04:33我想指出的是:
anicetomas的代码解释这里有点错误:

String   s1=new   String( "test ");//创建2个对象,一个Class和一个堆里面
String   s2= "test ";//创建1个对象,s2指向pool里面的 "test "对象



回复于:2008-06-01 09:21:49是两个对象,但不是N多人所说的一个对象和一个指向该对象的引用,引用并不是对象。

在创建一个String s = new String("java");的时候,JVM会到constant pool也就是常量池中

去检查看是否有一个"java"对象了,如果没有则在常量池中创建一个。

之后会在堆内存中分配了一个空间,放置这个new出来的String对象,形式如下:java.lang.String@123b,

@后面的数字就是该对象的HashCode。

常量池是在编译期生成的,而new一个对象是在运行时进行的,有一个先后顺序。


分享到:
评论

相关推荐

    String对象的内存分析

    对于`c`,`new String("xyz")`会创建一个新的`String`对象在堆中。同样,`d`也会在堆中创建一个新的`String`对象,尽管常量池中已有"xyz"。 3. `String s1 = new String("xyz"); String s2 = new String("xyz"); ...

    Java问题宝典2012版

    创建了几个String Object? 二者之间有什么区别? 25 34、String 和StringBuffer的区别 25 35、如何把一段逗号分割的字符串转换成一个数组? 26 36、数组有没有length()这个方法? String有没有length()这个方法? 26 ...

    JAVA面试题String产生了几个对象

    这个问题是很多Java开发者面试中常见的问题,本文将通过示例代码详细介绍String s = new String("xyz");产生了几个对象。 首先,需要了解Java中的字符串常量池的工作原理。Java设计了一个字符串常量池,以避免产生...

    java面试题

    第一,谈谈final, finally, finalize的区别。 第二,Anonymous Inner Class (匿名内部类) 是否可以extends...创建了几个String Object? 第十,Math.round(11.5)等於多少? Math.round(-11.5)等於多少?

    java 面试常问的问题 如何回答

    创建了几个String Object? 二者之间有什么区别? 23 34、String 和StringBuffer的区别 23 35、如何把一段逗号分割的字符串转换成一个数组? 24 36、数组有没有length()这个方法? String有没有length()这个方法? 24 ...

    2022年JAVA面试题及答案.doc

    JAVA 面试题及答案.doc 该资源提供了一份 JAVA 面试题集,涵盖了 JAVA 语言的多个...* 该语句创建了两个 String 对象:一个是 String 常量池中的“xyz”,另一个是new String("xyz") 创建的对象。 ...(以下省略)

    神州租车面试题

    创建了几个String Object? 分析:这道题考察了String对象的创建机制。在Java中,String对象可以通过new关键字创建,也可以通过字面值的方式创建。当我们使用new关键字创建String对象时,JVM会在堆中创建一个新的...

    java_ms.rar_Math Class_java collection

     第九,String s = new String("xyz") 创建了几个String Object?   第十,Math.round(11.5)等於多少? Math.round(-11.5)等於多少?   第十一,short s1 = 1 s1 = s1 + 1 有什么错? short s1 = 1 s1 += 1 有什么...

    java各大软件公司的面试题的集和

    创建了几个String Object? 第十,Math.round(11.5)等於多少? Math.round(-11.5)等於多少? 第十一,short s1 = 1; s1 = s1 + 1;有什么错? short s1 = 1; s1 += 1;有什么错? 面试题都是很变态的,要做好...

    C#面试笔试题目总结

    创建了几个 String Object? 两个对象。一个是 xyz,一个是指向 xyz 引用的 s。 22. GC 是什么?为什么要有 GC? GC 是垃圾回收器。因为有了 GC 就省去了程序员手工编码释放内存。 23. 能用 foreach 遍历访问的...

    java笔试题目以及部分答案

    创建了几个 String Object? String s = new String("xyz"); 创建了两个 String Object,一个是字符串常量池中的对象,另一个是堆中的对象。 10. Math.round(11.5) 等於多少? Math.round(-11.5) 等於多少? ...

    2012-Java面试宝典new版

    `创建了几个`String`对象? 二者之间有什么区别? **答案**: 这行代码创建了两个`String`对象:一个是堆内存中的新对象,另一个是字符串常量池中的对象。“xyz”这个字符串文本在字符串常量池中只存在一份,而`s`...

    Java面试宝典2017.zip

    一. Java基础部分 7 1、一个".java"源文件中是否可以包括多个类(不是内部类)?有什么限制? 7 2、Java有没有goto?...创建了几个String Object? 二者之间有什么区别? 23 34、String 和StringBuffer的区别

    中软国际Java程序员笔试题及答案.doc

    创建了几个 String Object? * 两个对象,一个是“xyz”,一个是指向“xyz”的引用对象 s。 12. Math.round(11.5) 等於多少?Math.round(-11.5) 等於多少? * Math.round(11.5) 返回 12,Math.round(-11.5) 返回...

    Java笔试题(派诚)[借鉴].pdf

    - **问题**: `String s=new String("xyz");` 创建了几个对象? - **解答**: 这条语句创建了**两个对象**: - 一个位于字符串常量池中的字符串 `"xyz"`。 - 一个通过 `new String()` 构造方法创建的新对象,该对象...

    java面试宝典

    Strings=newString(“xyz”);创建了几个String Object? 二者之间有什么区别?** 这行代码创建了一个`String`对象。通过`new String("xyz")`方式创建的字符串,会在堆内存中创建一个对象,并且字符串常量池中也会...

    Java应届生面试题1.doc

    创建了两个 String 对象:一个是字符串常量池中的“xyz”,另一个是堆中 new 出来的 String 对象。 6. String 和 StringBuffer、StringBuilder 之间区别。(5 分) String 是不可变的,StringBuffer 和 ...

    Java基础方面49273.doc

    4. **String对象创建**:`String s = new String("xyz");`这行代码会创建两个对象,一个是字符数组"xyz",另一个是String对象引用这个字符数组。 5. **Math.round方法**:`Math.round(11.5)`返回12,`Math.round(-...

    Java 应届生面试题

    * 创建了几个 String 对象?如果 String 常理池中已经创建了该对象,则不会继续创建,只创建了一个对象 new String("xyz");否则,将创建两个对象。 六、String、StringBuffer 和 StringBuilder 之间的区别 * 执行...

Global site tag (gtag.js) - Google Analytics