A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 杨卓儒 黑马帝   /  2012-8-25 17:22  /  2575 人查看  /  9 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

本帖最后由 杨卓儒 于 2012-8-25 23:34 编辑

String s = "Hello";s = s + " world!";这两行代码执行后,原始的String对象中的内容到底变了没有?

最好能透彻的分析一下~~~~~~~

9 个回复

倒序浏览
字符串本身就是一个对象,一旦初始化就不能被改变,之所以执行之后是“Hello world”,是因为s在内存中的指向变了。
例如,s1="abc"; s2="abc";  System.out.println(s1==s2);  结果是true,用==比较的时候,不但比较内容,还比较内存地址,
所以s1和s2在内存中指向的都是那一个"abc"对象

评分

参与人数 1技术分 +1 收起 理由
张_涛 + 1 赞一个!

查看全部评分

回复 使用道具 举报
String s = "Hello"; //s是一个引用,指向字符串对象"Hello",存储的是"Hello"对象的地址值
s = s + " world!"; // s + " world!"是创建了一个新的字符串对象"Hello world!",然后把新字符串对象的地址值赋给s,即s不再指向字符串对象"Hello",指向的是对象"Hello world!"
原来的对象"Hello"会在没有引用指向它的时候,被垃圾回收器回收。

评分

参与人数 1技术分 +1 收起 理由
张_涛 + 1 赞一个!

查看全部评分

回复 使用道具 举报
以下是张老师曾经总结的,顺便默哀2分钟
   
没有。因为String被设计成不可变(immutable)类,所以它的所有对象都是不可变对象。在这段代码中,s原先指向一个String对象,内容是 "Hello",然后我们对s进行了+操作,那么s所指向的那个对象是否发生了改变呢?答案是没有。这时,s不指向原来那个对象了,而指向了另一个 String对象,内容为"Hello world!",原来那个对象还存在于内存之中,只是s这个引用变量不再指向它了。
通过上面的说明,我们很容易导出另一个结论,如果经常对字符串进行各种各样的修改,或者说,不可预见的修改,那么使用String来代表字符串的话会引起很大的内存开销。因为 String对象建立之后不能再改变,所以对于每一个不同的字符串,都需要一个String对象来表示。这时,应该考虑使用StringBuffer类,它允许修改,而不是每个不同的字符串都要生成一个新的对象。并且,这两种类的对象转换十分容易。
同时,我们还可以知道,如果要使用内容相同的字符串,不必每次都new一个String。例如我们要在构造器中对一个名叫s的String引用变量进行初始化,把它设置为初始值,应当这样做:
public class Demo {
private String s;
...
public Demo {
s = "Initial Value";
}
...
}
而非
s = new String("Initial Value");     

     后者每次都会调用构造器,生成新对象,性能低下且内存开销大,并且没有意义,因为String对象不可改变,所以对于内容相同的字符串,只要一个String对象来表示就可以了。也就说,多次调用上面的构造器创建多个对象,他们的String类型属性s都指向同一个对象。
上面的结论还基于这样一个事实:对于字符串常量,如果内容相同,Java认为它们代表同一个String对象。而用关键字new调用构造器,总是会创建一个新的对象,无论内容是否相同。

     至于为什么要把String类设计成不可变类,是它的用途决定的。其实不只String,很多Java标准类库中的类都是不可变的。在开发一个系统的时候,我们有时候也需要设计不可变类,来传递一组相关的值,这也是面向对象思想的体现。不可变类有一些优点,比如因为它的对象是只读的,所以多线程并发访问也不会有任何问题。当然也有一些缺点,比如每个不同的状态都要一个对象来代表,可能会造成性能上的问题。所以Java标准类库还提供了一个可变版本,即 StringBuffer。

评分

参与人数 1技术分 +1 收起 理由
张_涛 + 1 赞一个!

查看全部评分

回复 使用道具 举报
唐见 来自手机 中级黑马 2012-8-25 17:40:51
报纸
java中所有字符串字面值都是一个String对象,String类在java中被定义成了final。所以你是说的原始的String对象中的内容是不会改变的。只是引用s的指向发生了变化。

评分

参与人数 1技术分 +1 收起 理由
张_涛 + 1 赞一个!

查看全部评分

回复 使用道具 举报
public class Demo{

    public static void main(String [] args){
            String s = "Hello";//这句话中,你创建了一个对象,栈中是String s,堆中是"Hello。
            s = s + " world!";//这里你又创建了一个对象Hello world!,你将栈中s地址指向了新的堆中Hello world!。而原来Hello这个对象还在堆中。
            System.out.println(s);//打印的是Hello world!,因为选择栈中的S执向的是对象Hello world的地址。
    }
}
回复 使用道具 举报
String s = "Hello";s = s + " world!";

   String类型数据是引用型数据,也就是说 变量名 s中存的是值"Hello"的内存地址,当使用字符串 "Hello"时,实际是引用了 s中的内存地址中的内容。当s = s + " world!";时,  实际上,是在内存中又创建了个
" Helloworld!"对象,这时, " Helloworld!"在内存中的地址值赋给了  s ,再使用 s 时,是使用了 s 中存储的内存地址中的值。
  "Hello"是 s赋值前引用的对象,
" Helloworld!" 是s = s + " world!"后重新创建的对象,

       引用型变量或者数据改变的是 变量名中所存的数据的地址。所以原始的String对象中并没有变,只是s不再引用它了。


回复 使用道具 举报
吴通 中级黑马 2012-8-25 22:09:10
8#
Sting字符串的最大特点就是:一旦被初始化就不能被改变
改变的只是s的指针,而远来的Sting字符串并未改变
回复 使用道具 举报
杨康 中级黑马 2012-8-25 23:20:39
9#
String s = "hello";//在堆内存中创建了一个字符串对象hello,在栈内存中声明了一个变量s指向堆内存中创建出的对象。
s = s + "World"//s重新指向了堆内存中的一个新对象,此时s指向的对象变为 "hello World",原来的hello对象已经不被s指向,但是仍然存在于内存中,如果没有被使用,就会被垃圾回收器回收。
回复 使用道具 举报
谢谢大家指导  终于明白了~~~~~~~~~
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马