Java方法中使用的是值传递(pass-by-value)!

   实在看不下去网上的一些面试题,很多都是错的答案。例如像今天这个问题:java方法用的是值传递还是引用传递。你在blogjava上还能搜到不同的答案呢。最近有空就翻译了一篇国外的文章,很多东西不能只看答案,而不知其所以然。第一次翻译文章,博友多指教。
    重申:对于原始类型(primitive type也译为值类型),是通过拷贝一个相同的值传给java方法的参数的;而对于引用类型(reference type),就是对象,是通过拷贝一个相同的应用或地址传给java方法的参数的。业界都统称这是pass-by-value(值传递),这里是翻译一篇 国外的文章来说明为什么java中的值传递比较特别。
    java中值传递比较特别,也比较有争议,所以重要的是要理解它的原理。推荐Java程序员都去读《Effective java》。


    原文地址:http://java.sun.com/developer/JDCTechTips/2001/tt1009.html
    另一篇写的比较好的文章(英文):http://www.brpreiss.com/books/opus5/html/page590.html
    blogjava上前辈的文章:http://www.blogjava.net/zygcs/archive/2008/10/05/232438.html

实际参数是如何传递到 java 方法的

HOW ARGUMENTS ARE PASSED TO JAVA METHODS

Suppose you're doing some Java programming, and you have a simple program like this:

假如你正在编写 java 程序,然后你写了以下一个程序:

  1   public   class  CallDemo1 {
 2            static   void  f( int  arg1) {
 3               arg1  =   10 ;
 4           }
 5       
 6            public   static   void  main(String args[]) {
 7                int  arg1;
 8       
 9               arg1  =   5 ;
10       
11               f(arg1);
12       
13               System.out.println( " arg1 =  "   +  arg1);
14           }
15       }
16  

In the main method, the variable arg1 is given the value 5, and then passed as an argument to the method f . This method declares a parameter of the same name, arg1 , used to access the argument.

main 方法中,变量 arg1 赋值 5 ,然后作为实际参数传递到方法 f 。这个方法声明一个相同名字的形式参数 arg1 ,被用于访问实际参数

What happens when you run this simple program? The method f modifies the value of arg1 , so what value gets printed, 5 or 10? It turns out that 5 is the right answer. This implies that setting arg1 to 10 in method f has no effect outside of the method.

当你执行这个程序会发生什么事情呢?方法 f 修改了 arg1 的值,所以会打印什么呢, 5 还是 10 ?这里 5 是正确的答案。这就意味着在方法 f 中,将 arg1 设为 10 并不影响到方法的外部。

Why does it work this way? The answer has to do with the distinction between the pass-by-value and pass-by-reference approaches to passing arguments to a method. The Java language uses pass-by-value exclusively. Before explaining what this means, let's look at an example of pass-by-reference, using C++ code:

为什么呢?答案在于区分使用“值传递”方式和“引用传递”方式给方法传递参数。 Java 语言使用与众不同的“值传递”方式。在解释它的意思之前,让我们看看以下一个值传递的例子,这是用 C++ 写的。

  1   #include  < iostream >
 2       
 3        using   namespace  std;
 4       
 5        void  f( int  arg1,  int &  arg2)
 6       {
 7           arg1  =   10 ;
 8           arg2  =   10 ;
 9       }
10       
11        int  main()
12       {
13            int  arg1, arg2;
14       
15           arg1  =   5 ;
16           arg2  =   5 ;
17       
18           f(arg1, arg2);
19       
20           cout  <<   " arg1 =  "   <<  arg1  <<   "  arg2 =  " <<  arg2  <<  endl;
22       }
23  

Function f has two parameters. The first parameter is a pass-by-value parameter, the second a pass-by-reference one. When you run this program, the first assignment in function f has no effect in the caller (the main function). But the second assignment does, in fact, change the value of arg2 in main . The & in int& says that arg2 is a pass-by-reference parameter. This particular example has no equivalent in Java programming.

函数 f 有两个形参。第一个是值传递,第二个是引用传递。当你运行这个程序的时候,第一个参数赋值不影响调用者( main 函数)。但是第二个参数赋值却会,事实上,修改了 main arg2 的值。在 int& 中的 & 表示 arg2 是一个引用传递的形参。这个特别的例子与 java 有所不同

So what does pass-by-value actually mean? To answer this, it's instructive to look at some JVM1 bytecodes that result from the following two commands:

那值传递到底是什么意思 ? 要回答这个问题,有需要看一下运行以下两行命令后的 JVM 的字节码

    javac CallDemo1.java

    javap -c -classpath . CallDemo1

Here is an excerpt from the output:

这里是输出的片段:

  Method  void  f( int )

        
0  bipush  10

        
2  istore_0

        
3   return

    Method 
void  main(java.lang.String[])

        
0  iconst_5

        
1  istore_1

        
2  iload_1

        
3  invokestatic # 2   < Method  void  f( int ) >


In the main method, the instruction iconst_5 pushes the value 5 onto the operand stack of the Java virtual machine. This value is then stored in the second local variable ( arg1 , with args as the first local variable). iload_1 pushes the value of the second local variable onto the operand stack, where it will serve as an argument to the called method.

main 方法中, iconst_5 指令将值 5 压入 JVM 的操作数堆栈( operand stack )。然后这个值随后存储第二个本地变量( arg1 的值存在第一个本地变量处)。 iload_1 指令将第二个本地变量的值再压入操作数堆栈,作为被调用方法的形参。

Then the method f is invoked using the invokestatic instruction. The argument value is popped from the stack, and is used to create a stack frame for the called method. This stack frame represents the local variables in f , with the method parameter ( arg1 ) as the first of the local variables.

然后方法 f 通过 invokestatic 指令调用。形参的值出栈,被用于为被调用的方法创建一个栈框架( stack frame )。这个栈框架表示所有 f 中的所有本地变量,而 arg1 实参将作为栈框架内的第一个本地变量。

What this means is that the parameters in a method are copies of the argument values passed to the method. If you modify a parameter, it has no effect on the caller. You are simply changing the copy's value in the stack frame that is used to hold local variables. There is no way to "get back" at the arguments in the calling method. So assigning to arg1 in f does not change arg1 in main . Also, the arg1 variables in f and in main are unrelated to each other, except that arg1 in f starts with a copy of the value of arg1 in main. The variables occupy different locations in memory, and the fact that they have the same name is irrelevant.

这里的的意思是形式参数作为实际参数的拷贝,被传入方法当中。如果你(在方法体内)修改了一个实际参数,它并不影响调用者,你只是修改了栈框架内的拷贝而已。不可能在调用方法中“取回”实参的值。因此在 f 中给 arg1 赋值不会改变 main 中的 arg1 ,而且这两个 arg1 毫无关联,除了 main 中的 arg1 f 中的 arg1 的一个拷贝之外。这两个变量存在内存的不同区域,事实上虽然它们有同样的名字,但是毫不相干。

By contrast, a pass-by-reference parameter is implemented by passing the memory address of the caller's argument to the called function. The argument address is copied into the parameter. The parameter contains an address that references the argument's memory location so that changes to the parameter actually change the argument value in the caller. In low-level terms, if you have the memory address of a variable, you can change the variable's value at will.

对比值传递,引用传递的参数是指传递调用者实参的内存地址给被调用的函数。实参地址拷贝给形参。形参指向了实参的内存区域,所以对形参的修改同样会修改到实参。从底层的角度来说,你可以任意修改变量的值,只要你有该变量的内存地址。

The discussion of argument passing is complicated by the fact that the term "reference" in pass-by-reference means something slightly different than the typical use of the term in Java programming. In Java, the term reference is used in the context of object references. When you pass an object reference to a method, you're not using pass-by-reference, but pass-by-value. In particular, a copy is made of the object reference argument value, and changes to the copy (through the parameter) have no effect in the caller. Let's look at a couple of examples to clarify this idea:

关于参数传递的讨论会复杂化,这是因为以上引用传递中的术语“引用”与 Java 编程中使用的(术语)存在微妙的区别。在 java 中,术语“引用”用于对象所有引用的上下文。当你给方法传递一个对象引用,你并不是使用引用传递,更像是值传递。尤其是,这样会拷贝一份实际参数的对象引用,(通过形参)对该拷贝的修改不会影响到调用者。让我们用以下两个例子来证明这个观点。

 

  1   class  A {
 2  
 3            public   int  x;
 4  
 5           A( int  x) {
 6                this .x  =  x;
 7           }
 8  
 9            public  String toString() {
10                return  Integer.toString(x);
11           }
12  
13       }
14  
15        public   class  CallDemo2 {
16  
17            static   void  f(A arg1) {
18               arg1  =   null ;
19           }
20  
21            public   static   void  main(String args[]) {
22               A arg1  =   new  A( 5 );
23               f(arg1);
24               System.out.println( " arg1 =  "   +  arg1);
25           }
26  
27       }

In this example, a reference to an A object is passed to f . Setting arg1 to null in f has no effect on the caller, just as in the previous example. The value 5 gets printed. The caller passes a copy of the object reference value ( arg1 ), not the memory address of arg1 . So the called method cannot get back at arg1 and change it.

在这个例子中, A 的引用传递给 f。 arg1 设置为 null 不影响调用者,就像前一个例子一样。程序会打印 5 。调用者传递一份对象引用( arg1 )的拷贝,而不是 arg1 的内存地址。所以调用方法无法取回到(调用者的) arg1 变量并修改它。

Here's another example:

 

 1   class  A {
 2  
 3            public   int  x;
 4  
 5            public  A( int  x) {
 6                this .x  =  x;
 7           }
 8  
 9            public  String toString() {
10                return  Integer.toString(x);
11           }
12  
13       }
14  
15      
16  
17        public   class  CallDemo3 {
18  
19            static   void  f(A arg1) {
20               arg1.x  =   10 ;
21           }
22  
23            public   static   void  main(String args[]) {
24               A arg1  =   new  A( 5 );
25               f(arg1);
26               System.out.println( " arg1 =  "   +  arg1);
27           }
28  
29       }



What gets printed here is 10 . How can that be? You've already seen that there's no way to change the caller's version of arg1 in the called method. But this code shows that the object referenced by arg1 can be changed. Here, the calling method and the called method have an object in common, and both methods can change the object. In this example, the object reference ( arg1 ) is passed by value. Then a copy of it is made into the stack frame for f . But both the original and the copy are object references, and they point to a common object in memory that can be modified.

这个例子会打印 10 。为什么呢 ? 通过之前那个例子,你已经知道无法改变在调用方法中修改(实参) arg1 。但是这里的代码显示 arg1 能够被修改。这里调用方法和被调用方法有个共同的对象,两个方法都能修改这个对象。在这个例子中,对象的引用( arg1 )是值传递。他的一个拷贝被加入到 f 的栈框架。但是原来的和拷贝的对象都是对象引用,它们指向相同的内存区域中的对象,因此可以修改。

In Java programming, it's common to say things like "a String object is passed to method f " or "an array is passed to method g ." Technically speaking, objects and arrays are not passed. Instead, references or addresses to them are passed. For example, if you have a Java object containing 25 integer fields, and each field is 4 bytes, then the object is approximately 100 bytes long. But when you pass this object as an argument to a method, there is no actual copy of 100 bytes. Instead, a pointer, reference, or address of the object is passed. The same object is referenced in the caller and the called method. By contrast, in a language like C++, it's possible to pass either an actual object or a pointer to the object.

java 编程中,经常可以听到“一个 String 对象被传递到方法 f ”或者“一个数组被传到方法 g ”。从技术上来说,不是传递一些对象或数组,而是传递引用或地址。例如,如果你有一个 java 对象包含 25 个整型的属性,每个属性占 4 字节,那这个对象大约是占 100 字节。但是作为形式参数传递的时候,不会拷贝 100 字节。而是传递一个指针、引用或地址。这样调用者和被调用的方法都指向同一个对象。对比而言,与 C++ 类似的语言,它可以传递真实的对象,也可以传递该对象的指针。

What are the implications of pass-by-value? One is that when you pass objects or arrays, the calling method and the called method share the objects, and both can change the object. So you might want to employ defensive copying techniques, as described in the September 4, 2001 Tech Tip, "Making Defensive Copies of Objects "

那什么是隐含的值传递呢?其一是当你传递对象或数组,由调用方法和被调用方法共享,双方都可修改对象。因此你可能想请入保护性拷贝技术,就像 2001 9 4 日技术贴士“使用保护性拷贝技术。”

You can fix the case above, where the called method modifies an object, by making the class immutable. An immutable class is one whose instances cannot be modified. Here's how you to do this:

你可以 修正以上的例子,当被调用方法修改一个对象时,让这个类成为不可变的。一个不可变类是指其实例不可修改的类。这里教你怎样去定义它。

 

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->  1   final   class  A {
 2  
 3            private   final   int  x;
 4  
 5            public  A( int  x) {
 6                this .x  =  x;
 7           }
 8  
 9            public  String toString() {
10                return  Integer.toString(x);
11           }
12       }
13  
14      
15  
16        public   class  CallDemo4 {
17  
18            static   void  f(A arg1) {
19                // arg1.x = 10;
20           }
21  
22            public   static   void  main(String args[]) {
23               A arg1  =   new  A( 5 );
24               f(arg1);
25               System.out.println( " arg1 =  "   +  arg1);
26  
27           }
28  
29       }



The printed result is 5 . Now uncomment the modification of A in f and recompile the program. Notice that it results in a compile error. You have made A immutable, so it can't be legally modified by f .

打印结果是 5 ,现在去掉 CallDemo4 f 方法内的注释和重新编译。注意它会编译出错。因为你已经让 A 不可变了,所以 f 不能合法修改它。

Another implication of pass-by-value is that you can't use method parameters to return multiple values from a method, unless you pass in a mutable object reference or array, and let the method modify the object. There are other ways of returning multiple values, such as returning an array from the method, or creating a specialized class and returning an instance of it.

另一个隐含的值传递是你不能利用方法的形参在一个方法中返回多个值,除非你传入一个不可变对象引用或数组,然后让方法修改该对象。还有另外的方式返回多个值,例如从方法中返回一个数组,或者创建一个特别定义的类的实例。

For more information about how arguments are passed to Java Methods, see Section 1.8.1, Invoking a Method, and section 2.6.4, Parameter Values, in "The Java Programming Language Third Edition " by Arnold, Gosling, and Holmes. Also see item 13, Favor immutability, and item 24, Make defensive copies when needed, in "Effective Java Programming Language Guide " by Joshua

想知道更多关于 Java 方法中的实际参数的传递的信息,请看 Arnold, Gosling, and Holmes 的《 The Java Programming Language Third Edition 》中的 1.8.1 节“ Invoking a Method ”和 2.6.4 节“ Parameter Values ”。还有 Joshua 的《 Effective Java Programming Language Guide 条款 13 Favor immutability ”和 24 Make defensive copies when needed ”。

 

你可能感兴趣的:(java,jvm,编程,框架,F#)