黑马程序员技术交流社区

标题: 有关String 类的问题。 [打印本页]

作者: 丁永志    时间: 2013-8-5 09:50
标题: 有关String 类的问题。
本帖最后由 杨兴庭 于 2013-8-7 18:05 编辑

今天看过视频之后还是有点迷糊。。求解答


String name=new String("
Hello"+"World!");

有几个对象这里是创建了几个对象?有点晕。
求解答,代点内存方面的回答就更好了。谢谢。






作者: 心灵之歌    时间: 2013-8-5 10:08
这里创建了俩个对象。1.new String();是一个对象。2.在new的同时,字符串在方法区的字符池里面也保留了一份字符串的对象。
作者: wang100753    时间: 2013-8-5 10:25
看例子:

例子A:
   String str1 = "java";
   String str2 = "java";
   System.out.print(str1==str2);
地球上有点Java基础的人都知道会输出false,因为==比较的是引用,equals比较的是内容。不是我忽悠大家,你们可以在自己的机子上运行一 下,结果是true!原因很简单,String对象被放进常量池里了,再次出现“java”字符串的时候,JVM很兴奋地把str2的引用也指向了 “java”对象,它认为自己节省了内存开销。不难理解吧 呵呵

例子B:
   String str1 = new String("java");
   String str2 = new String("java");
   System.out.print(str1==str2);
看过上例的都学聪明了,这次肯定会输出true!很不幸,JVM并没有这么做,结果是false。原因很简单,例子A中那种声明的方式确实是在 String常量池创建“java”对象,但是一旦看到new关键字,JVM会在堆中为String分配空间。两者声明方式貌合神离,这也是我把“如何创 建字符串对象”放到后面来讲的原因。大家要沉住气,还有一个例子。

例子C:
   String str1 = "java";
   String str2 = "blog";
   String s = str1+str2;
   System.out.print(s=="javablog");
再看这个例子,很多同志不敢妄言是true还是false了吧。爱玩脑筋急转弯的人会说是false吧……恭喜你,你会抢答了!把那个“吧”字去掉你就完 全正确。原因很简单,JVM确实会对型如String str1 = "java"; 的String对象放在字符串常量池里,但是它是在编译时刻那么做的,而String s = str1+str2; 是在运行时刻才能知道(我们当然一眼就看穿了,可是Java必须在运行时才知道的,人脑和电脑的结构不同),也就是说str1+str2是在堆里创建的, s引用当然不可能指向字符串常量池里的对象。没崩溃的人继续看例子D。

例子D:
   String s1 = "java";
   String s2 = new String("java");
   System.out.print(s1.intern()==s2.intern());
intern()是什么东东?反正结果是true。如果没用过这个方法,而且训练有素的程序员会去看JDK文档了。简单点说就是用intern()方法就 可以用“==”比较字符串的内容了。在我看到intern()方法到底有什么用之前,我认为它太多余了。其实我写的这一条也很多余,intern()方法 还存在诸多的问题,如效率、实现上的不统一……

例子E:
   String str1 = "java";
   String str2 = new String("java");
   System.out.print(str1.equals(str2));
无论在常量池还是堆中的对象,用equals()方法比较的就是内容,就这么简单!

作者: 黑马王晓明    时间: 2013-8-7 00:19
如果是
String name=new String("Hello");的话
我知道是创建了两个对象 new String 是一个 “Hello”也是一个对象  因为 "".方法()    是可以调用String类里面的方法的
String name=new String(“Hello”); 两个对象
String name="Hello";  一个对象  
至于你说的那个 我也在这请教高手指点一下……  
作者: 一只文字    时间: 2013-8-8 10:54
看看我这个例子吧  应该你就明白了

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个对象





欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2