Java开发网 Java开发网
注册 | 登录 | 帮助 | 搜索 | 排行榜 | 发帖统计  

您没有登录

» Java开发网 » 技术文章库  

按打印兼容模式打印这个话题 打印话题    把这个话题寄给朋友 寄给朋友    该主题的所有更新都将Email到你的邮箱 订阅主题
flat modethreaded modego to previous topicgo to next topicgo to back
作者 [转贴]Java 应用程序中的按值传递语义以及相关评论
iforem



发贴: 0
积分: 0
于 2003-12-12 13:09 user profilesend a private message to usersearch all posts byselect and copy to clipboard. 
ie only, sorry for netscape users:-)add this post to my favorite list
Java 应用程序中的按值传递语义
cherami 转贴
http://www.javaresearch.org

节选理解参数是按值而不是按引用传递的说明 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 %d\n", val); printf("pint is %d\n", pint); printf("*pint is %d\n", *pint); printf("ref is %d\n\n", ref); printf("calling modify\n"); //按值传递 val 和 pint,按引用传递 ref。 modify(val, pint, ref); printf("returned from modify\n\n"); printf("val is %d\n", val); printf("pint is %d\n", pint); printf("*pint is %d\n", *pint); printf("ref is %d\n", 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 %d\n", a); printf("p is %d\n", p); printf("r is %d\n", r);}

这段代码的输出为:

清单 2:C++ 代码的输出

val is 10pint is 4262128*pint is 15ref is 50calling modifyin modify...a is 0p is 0r is 0returned from modifyval is 10pint is 4262128*pint is 7ref 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 10sb1 is applessb2 is pearscalling modifyin modify...a is 0r1 is nullr2 is pears taste goodreturned from modifyval is 10sb1 is applessb2 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 中只有一种参数传递模式 -- 按值传递 -- 这有助于使事情保持简单。”

版权声明 本篇文章对您是否有帮助? 投票: 是 否 投票结果: 2 0







评论人:ljumwzc 参与分: 46 专家分: 40 发表时间: 2002-8-24 上午11:07
愚认为Java中是对象参数是按引用(地址)传递的,请看下例:

class august
{
int a;
StringBuffer sb1;
august( )
{
a=10;
sb1=new StringBuffer("你好,");
}

}

public class saturday
{
public static void method( august arg1)
{
arg1.a+=3;
arg1.sb1.append("Java");
System.out.println(arg1.a);
System.out.println(arg1.sb1);
}
public static void main(String args[])
{
august Agu=new august();
method(Agu);
System.out.println(Agu.a);
System.out.println(Agu.sb1);
}

}

如果按值传递,那么输出应该是:
13
你好,Java
10
你好,

但实际在j2sdk1.4.0中输出却是:
13
你好,Java
13
你好,Java

请问,做何解释?


评论人:ljumwzc 参与分: 46 专家分: 40 发表时间: 2002-8-24 上午11:09
愚认为Java中是对象参数是按引用(地址)传递的,请看下例:

class august
{
int a;
StringBuffer sb1;
august( )
{
a=10;
sb1=new StringBuffer("你好,");
}

}

public class saturday
{
public static void method( august arg1)
{
arg1.a+=3;
arg1.sb1.append("Java");
System.out.println(arg1.a);
System.out.println(arg1.sb1);
}
public static void main(String args[])
{
august Agu=new august();
method(Agu);
System.out.println(Agu.a);
System.out.println(Agu.sb1);
}

}

如果按值传递,那么输出应该是:
13
你好,Java
10
你好,

但实际在j2sdk1.4.0中输出却是:
13
你好,Java
13
你好,Java

请问,做何解释?


评论人:JFML 参与分: 616 专家分: 500 来自: 杭州
发表时间: 2002-8-24 下午7:23
你看看你传入method方法的参数

都不是基本类型,而是Object

所以,传入方法的其实是Ojbect引用的副本

举个例子,

我把我的银行账号(这是一个Object)给了张三

又把我的银行账号(Object副本)给了李四

张三往我的银行账号存了100$,李四也往我的银行帐号存了100$

那么你说我的银行存款现在有多少呢?

100$ ?

200$ ?

Understand ?


评论人:pinky 参与分: 138 专家分: 0 发表时间: 2002-8-26 下午3:17
cherami,你认为 JFML的例子是否恰当。
如果对副本的改变会传回副本,从而影响对象,那样的话岂不是也实现了传引用的同样的效果


评论人:JFML 参与分: 616 专家分: 500 来自: 杭州
发表时间: 2002-8-26 下午8:01
请仔细看看

你传入的参数是一个object

但是你在方法体内修改的不是object,而是object引用指向的一个成员变量

这就是关键所在

你体会一下两者的区别

class TTT
{
String s;
public TTT(String str)
{
s = str;
}
}
......

TTT t = new TTT("AAA"); // 1

method(TTT t)
{
t = new TTT("BBB"); // 2
}

// 3

这三个地方t.s的值分别是什么?

然后再看看下面

TTT t = new TTT("AAA"); // 1

method(TTT t)
{
t.s = "BBB"; // 2
}

// 3

那么这三个地方的t.s的值分别又是什么呢?


评论人:pinky 参与分: 138 专家分: 0 发表时间: 2002-8-27 下午12:01
你的例子我试了一下,第一种的结果是AAA BBB AAA,第二种的结果是AAA BBB BBB
这个结果也让我想清楚一些问题。对象引用的副本也是一个“指针“,他和对象的引用指向同一个区域。第一种情况下由于使用了new 语句,使副本获得一个新的指针,它指向的已不是原来的那个区域。第二种情况我还是很迷惑,因为它也实现了和c++里面一样的传引用的效果。可能我还要一段时间来想清楚这个问题。。。



评论人:JFML 参与分: 616 专家分: 500 来自: 杭州
发表时间: 2002-8-27 下午12:49
我不懂C++

所以当时理解地还比较快:)


评论人:pinky 参与分: 138 专家分: 0 发表时间: 2002-8-28 下午3:23
看了thinking in java 我对这个问题又有了新的认识。
传引用不一定传的就是副本,除非默认为自动生成副本
副本可以用clone生成,你也可以把自己的类创建成只读类,
或者为每个类都创建一个名为duplicate()的函数。
否则,否则不传副本,在调用方法内对对象引用的修改都最终会修改对象。
创建本地副本的目的就是防止原对象被修改。
因为能力有限,我不能把这个问题阐述的很清楚,只是提出一个观点,希望cherami
能够再考虑一下这个问题,帮助大家把它搞清楚



评论人:jigsaw 参与分: 72 专家分: 10 发表时间: 2002-11-14 下午6:05
How to Create a Reference to a Primitive
This is a useful technique if you need to create the effect of passing primitive
values by reference. Simply pass an array of one primitive element over
the method call, and the called method can now change the value seen by
the caller. Like this:

public class PrimitiveReference {
public static void main(String args[]) {
int [] myValue = { 1 };
modifyIt(myValue);
System.out.println(“myValue contains “ +
myValue[0]);
}
public static void modifyIt(int [] value) {
value[0]++;
}
}
//这是从sybex的complete guide里面抄来的。


评论人:kert 参与分: 558 专家分: 470 发表时间: 2002-11-15 下午1:34
我的理解:

我认为Java中参数的确是按值传递,然而应该如何理解参数所传递的指呢?
我觉得可以先解释一下在Java中对象和对象的引用之间的关系!
我们看一下的一个语句:

Integer i = new Integer(1);
在这里事实上发生了两件事,

JVM为i分配了一个引用存放在堆栈中(暂时我们称为引用refI)
JVM实例化了对象i,在队Heap中为其分配了一个存储空间用来存放数据。并且将refI指向这个堆中的内存块。

因此通常我们的传递的参数都只是这个refI,而非真正的i。

因为Java是值传递,所以传递的refI是无法被改变得。而同C一样,我们可以改变refI说指向的那些数据。

public class TempTest { static class MyInteger { public MyInteger(int inner) { this.inner = inner; } private int inner; public void setInt(int i) { inner = i; } public int getInt() { return inner; } public String toString() { return "" + inner; } } public static void swap(MyInteger a, MyInteger b) { pl("swap"); MyInteger temp = a; a = b; b = temp; } public static void swapII(MyInteger a, MyInteger b) { pl("swapII"); int temp = a.getInt(); a.setInt(b.getInt()); b.setInt(temp); } public static void swapI(MyInteger a, MyInteger b) { pl("swapI"); MyInteger temp = a; a = new MyInteger(b.getInt()); b = new MyInteger(temp.getInt()); } public static void main(String[] args) { MyInteger a = new MyInteger(1); MyInteger b = new MyInteger(2); plAngel; plBeer; swap(a,b); plAngel; plBeer; swapI(a,b); plAngel; plBeer; swapII(a,b); plAngel; plBeer; } private static void pl(Object a) { System.out.printlnAngel; }}

因此上面的例子的输出为

12swap12swapI12swapII21




话题树型展开
人气 标题 作者 字数 发贴时间
6856 [转贴]Java 应用程序中的按值传递语义以及相关评论 iforem 11726 2003-12-12 13:09

flat modethreaded modego to previous topicgo to next topicgo to back
  已读帖子
  新的帖子
  被删除的帖子
Jump to the top of page

   Powered by Jute Powerful Forum® Version Jute 1.5.6 Ent
Copyright © 2002-2021 Cjsdn Team. All Righits Reserved. 闽ICP备05005120号-1
客服电话 18559299278    客服信箱 714923@qq.com    客服QQ 714923