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

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© adf 初级黑马   /  2014-4-21 14:19  /  1115 人查看  /  9 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

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

9 个回复

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


而非
  1. s = new String("Initial Value");
复制代码


后者每次都会调用构造器,生成新对象,性能低下且内存开销大,并且没有意义,因为String对象不可改变,所以对于内容相同的字符串,只要一个String对象来表示就可以了。也就说,多次调用上面的构造器创建多个对象,他们的String类型属性s都指向同一个对象。
上面的结论还基于这样一个事实:对于字符串常量,如果内容相同,Java认为它们代表同一个String对象。而用关键字new调用构造器,总是会创建一个新的对象,无论内容是否相同。
至于为什么要把String类设计成不可变类,是它的用途决定的。其实不只String,很多Java标准类库中的类都是不可变的。在开发一个系统的时候,我们有时候也需要设计不可变类,来传递一组相关的值,这也是面向对象思想的体现。不可变类有一些优点,比如因为它的对象是只读的,所以多线程并发访问也不会有任何问题。当然也有一些缺点,比如每个不同的状态都要一个对象来代表,可能会造成性能上的问题。所以Java标准类库还提供了一个可变版本,即 StringBuffer

点评

分析很透彻,长见识了~!!!  发表于 2014-4-21 20:38
回复 使用道具 举报 1 0
我认为原始的String没有改变,原因:
String 类:适用于描述字符串常见事物。
String最大特点:一旦被初始化就不可以改变,改变的只是对象的应用s的指向变了。
回复 使用道具 举报
没有
s指向了一个新的对象“Hello world"
原有对象"Hello"未变,但因为没有引用会被回收的
回复 使用道具 举报
没有。String一旦创建对象,对象的内容是不会改变的。

你举的例子只是s改变了指向而已。hello是不会动的。
回复 使用道具 举报
没有。一旦定义它的长度就不可变。如下题:
String str = “abcdefg”;
str.substring(0,2);
  System.out.println(str);
程序的运行结果为abcdef,不会因为截取操作后而发生改变。
而与之相对应的StringBuffer其长度是可变的,如下:
public stati void main(String[] args){
StringBuffer sb = new StringBuffer();
sb.append("qq").append("ww");
show(sb,"ss");
System.out.println(sb.length());
        }
static void show(StringBuffer sb,String str){
          sb.append(str);
        }
程序输出结果为6,因为添加元素后其长度发生了改变。
回复 使用道具 举报
“hello”没有改变,而 s+"world" 是另一个新的String对象,s=s+"world"语句执行后是指s中存储的是“hello world”这个新对象的地址。
回复 使用道具 举报
String s = "hello"   在这条语句中,创建的对象 是String类型的 对象"hello",   s是String 类型的引用变量,引用变量s指向 了对象"hello",
s=s+"word" 这条语句中,你可以理解为”s+"word"“是一个新的String 类型对象"hello word",然后引用变量s指向了这个新的对象 hello word,而原来的对象"hello"依旧会存在,只不过s不再指向hello了
回复 使用道具 举报
第一次你写"Hello"的时候,他就已经存在于内存的一块区域(方法区)中了。  这块区域中的数据生命周期很长,没有对象指向他们的时候,他们也不会被垃圾回收机回收。
而String s = "Hello";只是把"Hello"的地址,告诉了s,s就指向了"Hello";
当s=s + "  World!"的时候,"  World!"也存在于内存的方法区中,s就指向了“Hello  World!”。
你指的对象的内容应该是说s的内容,s的指向现在变了。但是"Hello"肯定不会改变。
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马