首页 新闻 论坛 群组 Blog 文档 下载 读书 Tag 网摘 搜索 .NET Java 游戏 视频 人才 外包 培训 数据库 书店 程序员
中国软件网
欢迎您:游客 | 登录 注册 帮助
  • 参数传递问题,迷惑了。 [已结贴,结贴人:tujiazu]
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 18:14:32 楼主
    public class StringTester

    {  public static void changeString(String s,String s2, StringBuffer sb)

      {  s += " by Definition";
          s2="good";
          sb.append(" by Definition");
          sb.insert(4,"2");

      }

      public static void main(String args[])

      {  String string = new String("Java");
            String string2 = "good";

          StringBuffer buffer = new StringBuffer("Java");

          changeString(string,string2, buffer);

          System.out.println("String after method call: " + string);
         
          System.out.println("string2----->"+string2);
         
          System.out.println("StringBuffer after method call: " + buffer);

      }

    }


    /*运行结果:
    String after method call: Java
    string2----->good
    StringBuffer after method call: Java2 by Definition

    */

    方法体内改变了对象的内容,为何string,string2 打印的结果还是原来的值,为什么呢?
    20  修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • anqini
    • 等级:
    发表于:2008-06-08 18:17:291楼 得分:0
    String是不变的,当你想改变它的时候新创建一个,改变之后不会返回this的,
    StringBuffer 就不一样了
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 18:31:572楼 得分:0
    对于引用参数,也是把实参值复制给形参,只是这次复制的值是实参的栈空间中存放的值(栈中存了堆中的首地址值),也就是说:实参与形参将共用同一个堆空间。为此: 
      方法中如果改变了堆空间中的内容,实参能同步改动; 
      方法中如果改变了形参本身的栈中值(存放了一个新的堆空间首地址值),由于是值传递,实参是不变的 

      我明白了s,和s2,存放在栈内存中,他们存放的地址改变了,实参不变。只有堆内存改变了,实参跟着变。

    可是对于StringBuffer还是不理解。请教高手详解内部机制,先谢了。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 19:03:433楼 得分:0
    总共可以分为三步:
    1 <img src="http://cn.photos.yahoo.com/users/4694730ezbc109647/e451/__sr_/37d0.jpg?y1NhAcOvx1Go802ArDXT.A--&amp;F18&amp;amrzmtbyU8OakxxZ0_vGGw--" align=center alt="Image" />

    2 <img src="http://cn.photos.yahoo.com/users/4694730ezbc109647/e451/__sr_/4d83.jpg?P4arWexaDgP5Jgac8uH5Qg--&amp;F18&amp;amrzmtbyU8OakxxZ0_vGGw--" align=center alt="Image" />

    3 <img src="http://cn.photos.yahoo.com/users/4694730ezbc109647/e451/__sr_/f558.jpg?PqnTuYfW82_8EjWK1iNrkw--&amp;F18&amp;amrzmtbyU8OakxxZ0_vGGw--" align=center alt="Image" />
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 19:05:254楼 得分:0
    1.
    2.
    3.
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 19:08:405楼 得分:0
    1.
    2.
    3.
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • zhangkai08111
    • 等级:
    发表于:2008-06-08 19:21:016楼 得分:0
    3,4,5楼想干么。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 19:25:507楼 得分:0
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 19:31:068楼 得分:0
    1>java的参数传递,永远是传值
    2>String对象是immutable的
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 20:32:079楼 得分:0
        Stringbuffer 的内部机制,我还没有弄懂啊,高手帮忙啊。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 20:53:4110楼 得分:0
    String是不可变的,而StringBuffer是可变的。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 20:56:1011楼 得分:1
    也就是说String改变以后,原来的那个对象是不会变的,又重新生成了一个新对象,
    而StringBuffer实在原来的对象上面改变的
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • magic256
    • 等级:
    发表于:2008-06-08 21:10:3612楼 得分:5
    因为String是不支持拼接字符串的,所以在做+=的时候,其实质是new了一个string 对象出来,所以形参生成了新的地址,和实参无关了
    而StringBuffer支持用append()拼接,所以操作形参的时候其实是在操作实参,
    Java code
    public class Para1 { public static void main(String[] args) { StringBuffer sb = new StringBuffer("Hello "); System.out.println("Before change, sb = " + sb); changeData(sb); System.out.println("After changeData(n), sb = " + sb); } public static void changeData(StringBuffer strBuf) { //strBuf = new StringBuffer("Hi "); strBuf.append("World!"); } }

    把strBuf = new StringBuffer("Hi ");行注释掉和不注释掉将看到两个不同的结果注释的时候:
    Before change, sb = Hello
    After changeData(n), sb = Hello World!
    不注释掉的时候:
    Before change, sb = Hello
    After changeData(n), sb = Hello
    当不注释掉的时候并没有出现:
    Before change, sb = Hello
    After changeData(n), sb = Hi World!

    这说明了,在strBuf = new StringBuffer("Hi ");这行中,strBuf在内存中开辟了一个新的存储地址,不再和开始的sb指向同一个位置,所以,print出来的sb的值没有改变。这个时候的strBuf为"Hi World!"(可以在changeDate方法里面print出来),sb为"Hello"。

    所以,根据上面的概念,Java传递参数是属于按值传递,这种情况下参数的值就是对该对象的引用。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 21:48:5413楼 得分:5
      就你的例子而言:
      public class StringTester { 
      public static void changeString(String s,String s2, StringBuffer sb)
      { 
          //这里有3步:首先生成" by Definition"对象,然后s所指向的对象和" by Definition"拼接生成新的对象,最后s引用指向刚生成的对象.注意,s原来是指向new String("Java")对象的,这里又指向了新生成的对象,那么原来的对象呢?实际上new String("Java")对象仍然被main方法中的本地变量string所指向,这就是为什么你的打印结果还是原来的值!      s += " by Definition";
          //s2也是同样的道理"
          s2="good";
          //sb就不一样了,调用它的append方法,实际上是对它的char数组直接操作!因为sb和main方法里的buffer指向同一个对象,所以sb的变化会影响到buffer!
          sb.append(" by Definition");
          sb.insert(4,"2");

      }

      public static void main(String args[])
      { 
        //string引用指向new String("Java")对象   
        String string = new String("Java");
        //string2引用指向"good"对象
        String string2 = "good";
        //buffer引用指向new StringBuffer("Java")对象
          StringBuffer buffer = new StringBuffer("Java");

          changeString(string,string2, buffer);

          System.out.println("String after method call: " + string);
         
          System.out.println("string2----->"+string2);
         
          System.out.println("StringBuffer after method call: " + buffer);

      }
    }
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 22:11:2214楼 得分:0
    string,string2 是固定的,除非你对他们进行重新附直,否则是不会变的,但StringBuffer不是这样
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 22:35:5015楼 得分:0
    public static void changeString(String s,String s2, StringBuffer sb)
    这个方法不返回值,所做的修改只是针对它的方法体中的局部变量,
    在public static void main(String args[])方法中并没有对string 和string2做修改,请注意变量的命名。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 22:35:5916楼 得分:5
    public class StringTester

    {  public static void changeString(String s,String s2, StringBuffer sb)

      {  s += " by Definition";
          s2="good";
          sb.append(" by Definition");
          sb.insert(4,"2");

      }

      public static void main(String args[])

      {  String string = new String("Java");
            String string2 = "good";

          StringBuffer buffer = new StringBuffer("Java");

          changeString(string,string2, buffer);

          System.out.println("String after method call: " + string);
         
          System.out.println("string2----->"+string2);
         
          System.out.println("StringBuffer after method call: " + buffer);

      }

    }


    /*运行结果:
    String after method call: Java
    string2----->good
    StringBuffer after method call: Java2 by Definition

    */

    方法体内改变了对象的内容,为何string,string2 打印的结果还是原来的值,
      原因就very 简单
      string 是不变的,只能改变引用,才能对String ,从程序的角度来看,看到的String
      是可变的,但是不是这样的,

      这句话你晓不晓的,
     
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 23:08:3017楼 得分:0
    StringBuffer 就是用来弥补String的
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-08 23:53:5018楼 得分:0
    这里是传值和传引用
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 00:12:5119楼 得分:1
    这个问题跟StringBuffer没有直接关系。
    关键是重新赋值跟调用函数是两码事。
    Java的变量是引用类型的,但参数的传递仍然是值传递的语义,这两者是两码事,不要搞混。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 00:15:4120楼 得分:0
    String对象是一个常量,对String对象做任何操作都会改变内存中的位置。
    而StringBuffer是一个可变的字符串,对StringBuffer做操作是,永远也是指向同一个地址。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 08:08:5021楼 得分:1
    引用 2 楼 tujiazu 的回复:
    对于引用参数,也是把实参值复制给形参,只是这次复制的值是实参的栈空间中存放的值(栈中存了堆中的首地址值),也就是说:实参与形参将共用同一个堆空间。为此: 
      方法中如果改变了堆空间中的内容,实参能同步改动; 
      方法中如果改变了形参本身的栈中值(存放了一个新的堆空间首地址值),由于是值传递,实参是不变的 

      我明白了s,和s2,存放在栈内存中,他们存放的地址改变了,实参不变。只有堆内存改变了,实参跟…


    呵呵,你真明白了吗?你那段话是什么意思.我倒很不明白.
    以下:
    String虽是对象,但它是不可变的.既然你想改变string,string2 的值(LZ可不是改变对象的值,改变的是引用的值),那就在方法体内加个return 语句就好了.
    StringBuffer可变对象,方法体内改变了对象的内容,它也变了.
    代码:
    public class StringTester

    {  public static String changeString(String s,String s2, StringBuffer sb)

      {  s += " by Definition";
          s2="good";
          sb.append(" by Definition");
          sb.insert(4,"2");
          return s;
      }

      public static void main(String args[])

      {  String string = new String("Java");
            String string2 = "good";

          StringBuffer buffer = new StringBuffer("Java");

          string=changeString(string,string2, buffer);

          System.out.println("String after method call: " + string);

          System.out.println("string2----->"+string2);

          System.out.println("StringBuffer after method call: " + buffer);

      }

    }这个帖子有 用.
    http://topic.csdn.net/u/20080608/00/f6038c9c-e96b-4368-8bfd-cae914fbb1d2.html
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 08:39:2422楼 得分:0
    Java对原型的传递采用值拷贝, 而对对象类型采用指针拷贝.

    这里你用的对象String比较特殊, 你在changeString里对s1和s2的改变实际上生成了新的对象, 然后把新生成对象的地址指针赋给了他们.

    而StringBuffer并没有发生地址改变, 你对他的内容的操作都反应到堆中.你可以在改变StringBuffer之前添加一行
    stringBuffer = new StringBuffer(), 但跳出当前程序栈后, 回到main栈, 你会发现你对StringBuffer的改变也不会应用到那个stringBuffer对象中去.
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 09:58:1823楼 得分:0

          string=changeString(string,string2, buffer); 再加上return 返回的值赋给string ,那么string 原先指向堆内存值就改变了,
          如果 将string2=string=changeString(string,string2, buffer);  string2 是引用变量,那么它所引用的变量值在堆内存中也改变了。
          那么,string或string2,原先所指的对象就等待垃圾回收机制处理。如果将string=changeString(string,string2, buffer);注释掉,
          return 将不起任何作用。
       
          不知道我理解的对不对,请高手赐教。
         

    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 10:16:3624楼 得分:2
          string=changeString(string,string2, buffer); 再加上return 返回的值赋给string ,那么string原来指向的那个对象没有变,只不过是你的string指向一个新的对象而已!
          如果 将string2=string=changeString(string,string2, buffer);  string2 是引用变量,那么它所引用的变量值在堆内存中也没有改变,也只是string2指向了一个新的对象而已!
          那么,string或string2,原先所指的对象就等待垃圾回收机制处理,(前提是那2个对象没有被其他任何引用所指向).
    如果将string=changeString(string,string2, buffer);注释掉,return 将不起任何作用?啥意思啊...
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 11:06:4025楼 得分:0
    return 将不起任何作用?啥意思啊... 呵呵,我没有说清楚:如果注释掉string2=string=changeString(string,string2, buffer);  打印的结果还是实参string的值java,而不是java by by Definition,其实没啥意思啊。再请教高手:string或者string2都指向了新的对象,那么原先他们所指的对象呢,按照你的说法,string或string2仍然指向原来的酒对象,这样string 就指向两个对象了?如果我重新定义:string="AAAA",是不是又创建了一个新对象,string指向"AAAA",同时又指向"java by Definition",还指向"java" 呢,指向三个对象,似乎很迷惑?

    回复24楼。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • lisl2003
    • 等级:
    发表于:2008-06-09 11:48:0726楼 得分:0
    学习一下。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 12:31:0027楼 得分:0
    一个引用同时只能指向一个对象,一个对象可以同时被多个引用指向!
    如果将string=changeString(string,string2, buffer);注释掉
    无论你有没有注释,你的string,string2都只是指向原来的对象
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 12:34:5528楼 得分:0
    sorry,说错,如果没有注释掉,string指向新的对象,以前的那个对象不再被string所指向,string2则没有影响!
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 13:23:2429楼 得分:0
    13 楼说的非常对,在changeString方法中它的局部变量s,s2指向发生了改变,而main方法中的string ,string2
    指向没有因此而改变,在changeString方法结束后s,s2消失,他们指向的对象稍后也被垃圾回收机制回收了。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 15:09:1030楼 得分:0
    学习
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 16:21:0631楼 得分:0
    LZ你基础知识非常欠缺,建议重新学习J2SE.
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    发表于:2008-06-09 16:34:1732楼 得分:0
    问题一:我声明了什么!

    String s = "Hello world!";

    许多人都做过这样的事情,但是,我们到底声明了什么?回答通常是:一个String,内容是“Hello world!”。这样模糊的回答通常是概念不清的根源。如果要准确的回答,一半的人大概会回答错误。
    这个语句声明的是一个指向对象的引用,名为“s”,可以指向类型为String的任何对象,目前指向"Hello world!"这个String类型的对象。这就是真正发生的事情。我们并没有声明一个String对象,我们只是声明了一个只能指向String对象的引用变量。所以,如果在刚才那句语句后面,如果再运行一句:

    String string = s;

    我们是声明了另外一个只能指向String对象的引用,名为string,并没有第二个对象产生,string还是指向原来那个对象,也就是,和s指向同一个对象。

    问题三:String到底变了没有?

    没有。因为String被设计成不可变(immutable)类,所以它的所有对象都是不可变对象。请看下列代码:

    String s = "Hello";
    s = s + " world!";

    s 所指向的对象是否改变了呢?从本系列第一篇的结论很容易导出这个结论。我们来看看发生了什么事情。在这段代码中,s原先指向一个String对象,内容是 "Hello",然后我们对s进行了+操作,那么s所指向的那个对象是否发生了改变呢?答案是没有。这时,s不指向原来那个对象了,而指