当前位置:  技术问答>java相关

讨论:关于一个PassbyValue的特殊问题?

    来源: 互联网  发布时间:2015-03-28

    本文导语:  一.What will be printed when following code runs? 1. class Foo{ 2. static void change(String s){ 3. s=s.replace('j','l'); 4. } 5. 6. public static void main(String args[]){ 7. String s="java"; 8. change(s); 9. System.out.println(s); 10. } 11. } A. Compile...

一.What will be printed when following code runs?
1. class Foo{
2. static void change(String s){
3. s=s.replace('j','l');
4. }
5.
6. public static void main(String args[]){
7. String s="java";
8. change(s);
9. System.out.println(s);
10. }
11. }
A. Compiler error
B. Will print lava
C. Runtime exception
D. Will print java

这道题目我会,选择D.

但是下面这道题目:
java Mystery Java Hello

class Mystery
{
  public static void main(String[] args)
  {
    Changer c=new Changer();
    c.method(args);
    System.out.println(args[0]+" "+args[1]);
  }
  
  static class Changer
  {
    void method(String[] s)
    {
      String temp=s[0];
      s[0]=s[1];
      s[1]=temp;
    }
  }
}

答案是Hello Java,我写出的Java Hello

我就做错了,我觉得s[0]是String型值的引用,在方法体修改引用是不会对外部产生影响的啊.

为什么我错了呢?

谁能讲讲道理?

|
看看下面的文章吧:
Java 应用程序中的按值传递语义
几个月前,developerWorks 发布了我的书 Practical Java 中的一些节选,该书是由 Addison-Wesley 出版的。首先我将利用 developerWorks 上的此栏目回答读者提出的一些问题,然后对有关这些节选的各种评论作一答复。

节选理解参数是按值而不是按引用传递的说明 Java 应用程序有且仅有的一种参数传递机制,即按值传递。写它是为了揭穿普遍存在的一种神话,即认为 Java 应用程序按引用传递参数,以避免因依赖“按引用传递”这一行为而导致的常见编程错误。

对此节选的某些反馈意见认为,我把这一问题搞糊涂了,或者将它完全搞错了。许多不同意我的读者用 C++ 语言作为例子。因此,在此栏目中我将使用 C++ 和 Java 应用程序进一步阐明一些事实。

要点 
读完所有的评论以后,问题终于明白了,至少在一个主要问题上产生了混淆。某些评论认为我的节选是错的,因为对象是按引用传递的。对象确实是按引用传递的;节选与这没有冲突。节选中说所有参数都是按值 -- 另一个参数 -- 传递的。下面的说法是正确的:在 Java 应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。

C++ 和 Java 应用程序中的参数传递 
Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。这是一个重要特性,正如随后的代码示例所示的那样。

在继续讨论之前,定义按值传递和按引用传递这两个术语是重要的。按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数,调用代码中的原始值也随之改变。

关于 Java 应用程序中参数传递的某些混淆源于这样一个事实:许多程序员都是从 C++ 编程转向 Java 编程的。C++ 既包含非引用类型,又包含引用类型,并分别按值和按引用传递它们。Java 编程语言有基本类型和对象引用;因此,认为 Java 应用程序像 C++ 那样对基本类型使用按值传递,而对引用使用按引用传递是符合逻辑的。毕竟您会这么想,如果正在传递一个引用,则它一定是按引用传递的。很容易就会相信这一点,实际上有一段时间我也相信是这样,但这不正确。

在 C++ 和 Java 应用程序中,当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。请注意,调用方法的对象引用和副本都指向同一个对象。这是一个重要区别。Java 应用程序在传递不同类型的参数时,其作法与 C++ 并无不同。Java 应用程序按值传递所有参数,这样就制作所有参数的副本,而不管它们的类型。

示例 
我们将使用前面的定义和讨论分析一些示例。首先考虑一段 C++ 代码。C++ 语言同时使用按值传递和按引用传递的参数传递机制:

清单 1:C++ 示例 #include 
#include 

void modify(int a, int *P, int &r);

int main (int argc, char** argv)
{
  int val, ref;
  int *pint;

  val = 10;
  ref = 50;
  pint = (int*)malloc(sizeof(int));
  *pint = 15;

  printf("val is %dn", val);
  printf("pint is %dn", pint);
  printf("*pint is %dn", *pint);
  printf("ref is %dnn", ref);

  printf("calling modifyn");
  //按值传递 val 和 pint,按引用传递 ref。
  modify(val, pint, ref);
  printf("returned from modifynn");

  printf("val is %dn", val);
  printf("pint is %dn", pint);
  printf("*pint is %dn", *pint);
  printf("ref is %dn", ref);

  return 0;
}

void modify(int a, int *p, int &r)
{
    printf("in modify...n");
    a = 0;
    *p = 7;
    p = 0;
    r = 0;
    printf("a is %dn", a);
    printf("p is %dn", p);
    printf("r is %dn", r);
}

 


这段代码的输出为:

清单 2:C++ 代码的输出 val is 10
pint is 4262128
*pint is 15
ref is 50

calling modify
in modify...
a is 0
p is 0
r is 0
returned from modify

val is 10
pint is 4262128
*pint is 7
ref is 0
 


这段代码声明了三个变量:两个整型变量和一个指针变量。设置了每个变量的初始值并将其打印出来。同时打印出了指针值及其所指向的值。然后将所有三个变量作为参数传递给 modify 函数。前两个参数是按值传递的,最后一个参数是按引用传递的。modify 函数的函数原型表明最后一个参数要作为引用传递。回想一下,C++ 按值传递所有参数,引用除外,后者是按引用传递的。

modify 函数更改了所有三个参数的值: 

将第一个参数设置为 0。 
将第二个参数所指向的值设置为 7,然后将第二个参数设置为 0。 
将第三个参数设置为 0。

将新值打印出来,然后函数返回。当执行返回到 main 时,再次打印出这三个参数的值以及指针所指向的值。作为第一个和第二个参数传递的变量不受 modify 函数的影响,因为它们是按值传递的。但指针所指向的值改变了。请注意,与前两个参数不同,作为最后一个参数传递的变量被 modify 函数改变了,因为它是按引用传递的。

现在考虑用 Java 语言编写的类似代码:

清单 3:Java 应用程序 class Test
{
  public static void main(String args[])
  {
    int val;
    StringBuffer sb1, sb2;

    val = 10;
    sb1 = new StringBuffer("apples");
    sb2 = new StringBuffer("pears");
    System.out.println("val is " + val);
    System.out.println("sb1 is " + sb1);
    System.out.println("sb2 is " + sb2);
    System.out.println("");

    System.out.println("calling modify");
    //按值传递所有参数
    modify(val, sb1, sb2);
    System.out.println("returned from modify");
    System.out.println("");

    System.out.println("val is " + val);
    System.out.println("sb1 is " + sb1);
    System.out.println("sb2 is " + sb2);
  }

  public static void modify(int a, StringBuffer r1,
                            StringBuffer r2)
  {
      System.out.println("in modify...");
      a = 0;
      r1 = null;  //1
      r2.append(" taste good");
      System.out.println("a is " + a);
      System.out.println("r1 is " + r1);
      System.out.println("r2 is " + r2);
  }
}
 


这段代码的输出为:

清单 4:Java 应用程序的输出 val is 10
sb1 is apples
sb2 is pears

calling modify
in modify...
a is 0
r1 is null
r2 is pears taste good
returned from modify

val is 10
sb1 is apples
sb2 is pears taste good
 


这段代码声明了三个变量:一个整型变量和两个对象引用。设置了每个变量的初始值并将它们打印出来。然后将所有三个变量作为参数传递给 modify 方法。

modify 方法更改了所有三个参数的值: 

将第一个参数(整数)设置为 0。 
将第一个对象引用 r1 设置为 null。 
保留第二个引用 r2 的值,但通过调用 append 方法更改它所引用的对象(这与前面的 C++ 示例中对指针 p 的处理类似)。

当执行返回到 main 时,再次打印出这三个参数的值。正如预期的那样,整型的 val 没有改变。对象引用 sb1 也没有改变。如果 sb1 是按引用传递的,正如许多人声称的那样,它将为 null。但是,因为 Java 编程语言按值传递所有参数,所以是将 sb1 的引用的一个副本传递给了 modify 方法。当 modify 方法在 //1 位置将 r1 设置为 null 时,它只是对 sb1 的引用的一个副本进行了该操作,而不是像 C++ 中那样对原始值进行操作。

另外请注意,第二个对象引用 sb2 打印出的是在 modify 方法中设置的新字符串。即使 modify 中的变量 r2 只是引用 sb2 的一个副本,但它们指向同一个对象。因此,对复制的引用所调用的方法更改的是同一个对象。

编写一个交换方法 
假定我们知道参数是如何传递的,在 C++ 中编写一个交换函数可以用不同的方式完成。使用指针的交换函数类似以下代码,其中指针是按值传递的:

清单 5:使用指针的交换函数 #include 
#include 

void swap(int *a, int *b);

int main (int argc, char** argv)
{
  int val1, val2;
  val1 = 10;
  val2 = 50;
  swap(&val1, &val2);
  return 0;
}

void swap(int *a, int *b)
{
  int temp = *b;
  *b = *a;
  *a = temp;
}

 


使用引用的交换函数类似以下代码,其中引用是按引用传递的:

清单 6:使用引用的交换函数 #include 
#include 

void swap(int &a, int &b);

int main (int argc, char** argv)
{
  int val1, val2;
  val1 = 10;
  val2 = 50;
  swap(val1, val2);
  return 0;
}

void swap(int &a, int &b)
{
  int temp = b;
  b = a;
  a = temp;
}
 


两个 C++ 代码示例都像所希望的那样交换了值。如果 Java 应用程序使用“按引用传递”,则下面的交换方法应像 C++ 示例一样正常工作:

清单 7:Java 交换函数是否像 C++ 中那样按引用传递参数 class Swap
{
  public static void main(String args[])
  {
    Integer a, b;

    a = new Integer(10);
    b = new Integer(50);

    System.out.println("before swap...");
    System.out.println("a is " + a);
    System.out.println("b is " + b);
    swap(a, b);
    System.out.println("after swap...");
    System.out.println("a is " + a);
    System.out.println("b is " + b);
  }

  public static void swap(Integer a, Integer b)
  {
    Integer temp = a;
    a = b;
    b = temp;
  }
}
 


因为 Java 应用程序按值传递所有参数,所以这段代码不会正常工作,其生成的输入如下所示: 

清单 8:清单 7 的输出 before swap...
a is 10
b is 50
after swap...
a is 10
b is 50
 


那么,在 Java 应用程序中如何编写一个方法来交换两个基本类型的值或两个对象引用的值呢?因为 Java 应用程序按值传递所有的参数,所以您不能这样做。要交换值,您必须用在方法调用外部用内联来完成。

结论 
我在书中包括该信息的意图并不是作琐细的分析或试图使问题复杂化,而是想警告程序员:在 Java 应用程序中假定“按引用传递”语义是危险的。如果您在 Java 应用程序中假定“按引用传递”语义,您就可能写出类似上面的交换方法,然后疑惑它为什么不正常工作。

我必须承认,在我第一次认识到 Java 应用程序按值传递所有参数时,我也曾表示怀疑。我曾一直假定因为 Java 应用程序有两种类型,所以他们按值传递基本类型而按引用传递引用,就像 C++ 那样。在转向 Java 编程之前我已用 C++ 编程好几年了,感觉任何其他事情似乎都不直观。但是,一旦我理解了发生的事情,我就相信 Java 语言按值传递所有参数的方法更加直观。The Java Programming Language,Second Edition 的作者,Ken Arnold 和 James Gosling 在 2.6.1 节中说得最好:“在 Java 中只有一种参数传递模式 -- 按值传递 -- 这有助于使事情保持简单。”

资源 

要查找生成有效 Java 代码的实用方法,请参阅 Peter Haggar 的 Practical Java Programming Language Guide。 
要了解如何有效地利用 Java 的构造、库和语言详细资料,请参阅 Ken Arnold 和 James Gosling 的 Java Programming Language,Second Edition。 
请阅读 developerWorks 上发布的 Practical Java 的其他节选。
关于作者 
Peter Haggar 是位于北卡罗莱纳州的 Research Triangle Park 的 IBM 的高级软件工程师。他有丰富的编程经验,工作过的领域包括开发工具、类库和操作系统。在 IBM,Peter 的工作领域是新出现的 Java 技术。他目前正致力于嵌入式 Java 并且是 IBM 实时 Java 引用实现的项目负责人。在众多的行业会议上,Peter 经常就有关 Java 技术发言。他已为 IBM 工作了 12 年以上,并获得 Clarkson 大学计算机科学理学士学位。可以通过 haggar@us.ibm.com 与他联系。

|
传递的既然是s[],那就是数组了,数组是传递引用的

|
s[0],s[1],... 应该看作是s 这个引用的内容。
在函数内改变s的内容,就如Patrick_DK说的"对引用的值的修改是会保留的"。
这个问题以前也有讨论过的...

|
数组是传地址的

    
 
 

您可能感兴趣的文章:

  • 参数传递的问题!(大家讨论讨论)
  • 哪位高手有兴趣跟我讨论讨论java中调用dll文件??小弟有些问题还是不很清楚??
  • 和Java版高手在线讨论代理服务器的问题,讨论者都有分。
  • Java 访问控制的问题(public,private,protected,(default))!讨论讨论!
  • 这两天本版人气不高,我来发个问题,有关互斥同步的。大家讨论讨论
  • 请问这里可以讨论MINIX的问题吗?
  • 讨论linux的发展前途与有无用处问题。。。。。。。。。。。
  • LINUX下JAVA要代替C..?常跟朋友讨论的一个问题
  • 讨论LUXIN安装问题
  • 大家来讨论一个最实际的问题:)
  • 请问一下,关于arm下的嵌入式linux驱动相关问题请教是应该发在哪个讨论区?
  • 高手请进!讨论一个问题。
  • 关纯DOS下的256色、32K色、64K色问题讨论
  • 有挑战性的问题,大家一起讨论
  • 大家讨论一下指针数组的问题!
  • 老问题新讨论
  • 哇考,为什么在一个帖子里不能回复30次以上啊?这样子怎么和人家讨论问题啊?
  • 我看到大家都在讨论分数的问题,我如何看到自己的分数呢?
  • 请使用QT编程的朋友到以下网址讨论QT中遇到的问题!!!!!!!
  • Java与IE的问题!大家来讨论!
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • Java 可以做拨号程序吗?我只是和大家讨论讨论 不必太认真
  • 欢迎高手来讨论:关于文件格式的大讨论
  • 用java开发一个基于Proxy(代理)的网络计费系统。有兴趣的来讨论讨论
  • 【讨论贴】gcc开发的时候有大家都有什么好的调试方法,来讨论下
  • 讨论讨论,当错误发生时,并用if语句测试出时,应该返回怎样的值
  • 一个面试,是“北京华胜六所”外包给风河(VxWorks)北京研发处,做linux内核开发,大家过来讨论讨论
  • 新建了个QQ群(软件与创业),希望有兴趣的朋友进来讨论讨论软件项目、产品、创业、管理、投资等(代码之外的)观点和想法
  • 用java做c/s结构可行吗???大家来讨论讨论,应该都会有收获。
  • 有没有人讨论value object模式
  • http://www.itpub.net 论坛更换数据库,速度更快,欢迎大家前去讨论!
  • 讨论“内存泄漏”
  • 很专业的问题请教J2EE高手!这是一个讨论区,有请各位对J2EE感兴趣的朋友参加讨论!
  • 大家一起讨论讨论,suse和ubuntu的区别,顺便散散分
  • 对大家很有意义的一个问题,建议大虾、菜鸟们都来讨论讨论#¥#·#¥·#%#¥%#¥%
  • 大家来讨论一下java 的发展前景
  • 谁会JAVA让我们共同学习和讨论JAVA
  • 100分大讨论:Servlet + var 模式 !
  • 欢迎大家一起来讨论:集群在J2EE中的实现。
  • 为什么在这里讨论编程的很少?
  • 讨论一下UNIX中的硬链接与符号链接


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3