通常,当程序员讨论传值和传引用时,他们是指语言的参数传递机制,c++同时支持这两种机制,因此,以前使用过c++的程序员开始好像不能确定的java是如何传参数的。java语言为了事情变得简单只支持参数传值的机制。<BR> java中的变量有两种类型:引用类型和原始类型。当他们被作为参数传递给方法时,他们都是传值的。这是一个非常重要的差别,下面的代码范例将说明这一点。<BR> 在继续前,我们有必要定义一下传值和传引用。传值意味着当参数被传递给一个方法或者函数时,方法或者函数接收到的是原始值的副本。因此,如果方法或者函数修改了参数,受影响的只是副本,原始值保持不变。<BR> 关于java中的参数传递的混乱是因为很多java程序员是从c++转变过来的。c++有引用和非引用类型的变量,并且分别是通过传引用和传值得。java语言有原始类型和对象引用,那么,按照逻辑,java对于原始类型使用传值而对引用是传引用的,就像c++一样。毕竟,你会想到如果你正在传递一个引用,那么它一定是传引用的。这是一个很诱惑人的想法,但是是错误的!<BR> 在c++和java中,当函数的参数不是引用时,你传递的是值得副本(传值)。但是对于引用类型就不同了。在c++中,当参数是引用类型,你传递的是引用或者内存地址(传引用),而在java中,传递一个引用类型的参数的结果只是传递引用的副本(传值)而非引用自身。这是一个非常重要的区别!java不考虑参数的类型,一律传递参数的副本。<BR> 仍然不信?如果java中是传引用,那么下面的范例中的swap方法将交换他们的参数。因为是传值,因此这个方法不是像期望的那样正常工作。<BR> <pre> class Swap { public static void main(String args[]) { Integer a, b; int i,j; a = new Integer(10); b = new Integer(50); i = 5; j = 9; System.out.println("Before Swap, a is " + a); System.out.println("Before Swap, b is " + b); swap(a, b); System.out.println("After Swap a is " + a); System.out.println("After Swap b is " + b); System.out.println("Before Swap i is " + i); System.out.println("Before Swap j is " + j); swap(i,j); System.out.println("After Swap i is " + i); System.out.println("After Swap j is " + j); } public static void swap(Integer ia, Integer ib) { Integer temp = ia; ia = ib; ib = temp; } public static void swap(int li, int lj) { int temp = li; li = lj; lj = temp; } } </pre>
上面程序的输出是: <BR> <PRE> Before Swap, a is 10 Before Swap, b is 50 After Swap a is 10 After Swap b is 50 Before Swap i is 5 Before Swap j is 9 After Swap i is 5 After Swap j is 9 </PRE> 因为swap方法接收到的是引用参数的副本(传值),对他们的修改不会反射到调用代码。<BR>
译者注:在传递引用和原始类型时还是有不同的,考虑以下的代码: <pre> class Change { public static void main(String args[]) { StringBuffer a=new StringBuffer("ok"); int i; i = 5; System.out.println("Before change, a is " + a); change(a); System.out.println("After change a is " + a); System.out.println("Before change i is " + i); change(i); System.out.println("After change i is " + i); } public static void change(StringBuffer ia) { ia.append(" ok?"); } public static void change(int li) { li = 10; } } </pre> 程序的输出为:<BR> <PRE> Before change, a is ok After change a is ok ok? Before change i is 5 After change i is 5 </PRE> ,即如果传递的是引用,那么可以修改引用对象的内容,这个改变会影响到原来的对象,而传递的如果是原始类型则不会有影响。这个也是造成误解的原因之一吧。
关于 Java 应用程序中参数传递的某些混淆源于这样一个事实:许多程序员都是从 C++ 编程转向 Java 编程的。C++ 既包含非引用类型,又包含引用类型,并分别按值和按引用传递它们。Java 编程语言有基本类型和对象引用;因此,认为 Java 应用程序像 C++ 那样对基本类型使用按值传递,而对引用使用按引用传递是符合逻辑的。毕竟您会这么想,如果正在传递一个引用,则它一定是按引用传递的。很容易就会相信这一点,实际上有一段时间我也相信是这样,但这不正确。
在 C++ 和 Java 应用程序中,当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。请注意,调用方法的对象引用和副本都指向同一个对象。这是一个重要区别。Java 应用程序在传递不同类型的参数时,其作法与 C++ 并无不同。Java 应用程序按值传递所有参数,这样就制作所有参数的副本,而不管它们的类型。
示例 我们将使用前面的定义和讨论分析一些示例。首先考虑一段 C++ 代码。C++ 语言同时使用按值传递和按引用传递的参数传递机制:
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); }
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); }
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里面抄来的。