Java中的值传递和引用传递

来源:互联网 发布:淘宝上怎么出售qq号码 编辑:程序博客网 时间:2024/06/02 03:00

Java中没有指针,所以也没有引用传递了,仅仅有值传递不过可以通过对象的方式来实现引用传递 。类似java没有多继承 但可以用多次implements 接口实现多继承的功能 。


值传递:方法调用时,实际参数把它的值传递给对应的形式参数,方法执行中形式参数值的改变不影响实际参 数的值。 
引用传递:也称为传地址。方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,方法执行中形式参数值的改变将会影响实际参数的值。 1 f; F, R4 V) f1 q& l

Java参数按值传递 



面试题:当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递? 
    答:是值传递。Java 编程语言只有值传递参数。当一个对象实例作为一个参数被传递到方法中时,参数的值就是该对象的引用一个副本。指向同一个对象,对象的内容可以在被调用的方法中改变,但对象的引用(不是引用的副本)是永远不会改变的。 % E; X+ O6 s  _$ g

------------------------------------------------------------- 

在 Java 应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。 + j$ k; e2 g& p) D
( b5 G: k1 L% F& b+ u
Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。 
  7 M7 W. n6 K. L+ Q
按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数的值,调用代码中的原始值也随之改变。如果函数修改了该参数的地址,调用代码中的原始值不会改变. 8 e) l& q% h8 p! x, G
  
当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。 : g! J$ w$ n7 J6 |6 g2 F
  

Java 应用程序按值传递参数(引用类型或基本类型),其实都是传递他们的一份拷贝.而不是数据本身.(不是像 C++ 中那样对原始值进行操作。)

例1:  
      
       1. //在函数中传递基本数据类型,     
       2. public class Test {     
       3.               
       4.     public static void change(int i, int j) {     5 W7 M. z9 [9 P" A  b4 `5 i
       5.         int temp = i;     0 J: p" x) E/ ?# }+ K' O
       6.         i = j;     9 P1 c: ?9 H. t1 s
       7.         j = temp;     ( ^2 F4 Q% _3 c3 H7 s6 f4 J* y
       8.     }     
       9.     
      10.     public static void main(String[] args) {     
      11.         int a = 3;     
      12.         int b = 4;     
      13.         change(a, b);     
      14.     
      15.         System.out.println("a=" + a);     
      16.         System.out.println("b=" + b);     
      17.     }     
      18. }     3 P4 m5 n, L( n7 Q1 @( b$ T
      19.     2 `* D! H1 Z/ I, s& z4 h, z
      20. 结果为:     
      21. a=3    $ ?+ I3 B+ Q  S
      22. b=4    % T& Z+ m7 l8 G* B% ^6 w# r
      23. 原因就是 参数中传递的是 基本类型 a 和 b 的拷贝,在函数中交换的也是那份拷贝的值 而不是数据本身;    . C# c# x! M) c) _7 ^3 g% \& M5 I

 例2:  

    Java代码 复制代码  
      
       1. //传的是引用数据类型     : j6 f5 ^2 s0 E) d1 w0 m6 P; A# Q
       2. public class Test {     1 K8 R; \- s/ i# p2 H6 v. T
       3.     # |$ y1 M( B- r* F# o- D7 |
       4.     public static void change(int[] counts) {     " `. l- k1 ^5 V8 Q. V
       5.         counts[0] = 6;     
       6.         System.out.println(counts[0]);     
       7.     }     
       8.     
       9.     public static void main(String[] args) {     
      10.         int[] count = { 1, 2, 3, 4, 5 };     
      11.         change(count);     $ r6 _% v% `/ O3 c
      12.     }     , k* B$ ]8 R4 Z0 H+ h. U! U# U
      13. }     
      14.     4 r( S1 F" T" ?" J1 @8 N
      15. 在方法中 传递引用数据类型int数组,实际上传递的是其引用count的拷贝,他们都指向数组对象,在方法中可以改变数组对象的内容。即:对复制的引用所调用的方法更改的是同一个对象。

例3:  
    Java代码 复制代码  + ?- M, J( e$ Y
      / b- w( Z% p+ t
       1. //对象的引用(不是引用的副本)是永远不会改变的     2 T( o6 v3 {* G0 ^6 f& M
       2. class A {     
       3.     int i = 0;     : U3 B, J0 Y! ^3 m  {4 D& n: f5 Q* `
       4. }     
       5.     
       6.     * c6 r, v  l' u- c( I% z
       7. public class Test {     
       8.     
       9.     public static void add(A a) {     
      10.         a = new A();    % }/ g9 x8 O0 {* S) P1 H; u、、、
      11.         a.i++;     5 z3 u3 o/ i0 N3 Y0 ?' `
      12.     }     
      13.          1 l. X2 L2 |4 p1 s3 p' D, y' V! {
      14.     public static void main(String args[]) {     
      15.         A a = new A();     , Y0 h! B7 q: F1 w$ M
      16.         add(a);     
      17.         System.out.println(a.i);     8 M) S: ?  H* N
      18.     }     5 B3 I9 r" C8 l* i2 p9 x; {
      19. }     
      20.     8 s$ l, p& e7 z" r
      21. 输出结果是0    3 m) J. r7 u. n% U" X$ A
      22. 在该程序中,对象的引用指向的是A ,而在change方法中,传递的引用的一份副本则指向了一个新的OBJECT,并对其进行操作。     
      23. 而原来的A对象并没有发生任何变化。 引用指向的是还是原来的A对象。   * ?) {' w5 s( `/ _- G% O! x

 例4:  
    String 不改变,数组改变  
    Java代码 复制代码    E% t' y8 o. r% ~! F) E
      
       1.     ( J& q4 Q6 Y5 `6 A4 b) I
       2. public class Example {     ) r2 \0 `- b* M* M& I$ Q. A
       3.     String str = new String("good");     
       4.     
       5.     char[] ch = { 'a', 'b', 'c' };     
       6.     7 V8 J3 ?+ s; r* Z* p/ ^/ W. h
       7.     public static void main(String args[]) {     
       8.         Example ex = new Example();     
       9.         ex.change(ex.str, ex.ch);     ! n+ {; m) u& l1 b0 f- j4 |
      10.         System.out.print(ex.str + " and ");     $ R* q9 N: L4 u5 B1 E0 b$ A9 |
      11.         System.out.println(ex.ch);     
      12.     }     1 V! m) U+ J0 l
      13.     # D8 u' w4 O+ Y
      14.     public void change(String str, char ch[]) {     
      15.         str = "test ok";     
      16.         ch[0] = 'g';     . E; \' `4 N) }
      17.     }     ' n6 V) C5 U3 F) ~. D. M
      18. }      
      19. 程序3输出的是 good and gbc.
      20. String 比较特别,看过String 代码的都知道,String 是 final的。所以值是不变的。 函数中String对象引用的副本指向了另外一个新String对象,而数组对象引用的副本没有改变,而是改变对象中数据的内容.  
     21. 对于对象类型,也就是Object的子类,如果你在方法中修改了它的成员的值,那个修改是生效的,方法调用结束后,它的成员是新的值,但是如果你把它指向一个其它的对象,方法调用结束后,原来对它的引用并没用指向新的对象。 





Java参数,不管是原始类型还是引用类型,传递的都是副本(有另外一种说法是传值,但是说传副本更好理解吧,传值通常是相对传址而言)。 : y9 d: `9 m* W* {. G
   如果参数类型是原始类型,那么传过来的就是这个参数的一个副本,也就是这个原始参数的值,这个跟之前所谈的传值是一样的。如果在函数中改变了副本的值不会改变原始的值. , F8 N4 R( H4 U2 w
    如果参数类型是引用类型,那么传过来的就是这个引用参数的副本,这个副本存放的是参数的地址如果在函数中没有改变这个副本的地址,而是改变了地址中的值,那么在函数内的改变会影响到传入的参数。如果在函数中改变了副本的地址,如new一个,那么副本就指向了一个新的地址,此时传入的参数还是指向原来的地址,所以不会改变参数的值。 ; D9 U  D& f' G3 ?4 Q1 x6 \" }0 K- s
( 对象包括对象引用即地址和对象的内容) , }# m9 F" ^( J( i
/ S. z6 N8 D) B4 h. g" Z$ j
a.传递值的数据类型:八种基本数据类型和String(这样理解可以,但是事实上String也是传递的地址,只是string对象和其他对象是不同的,string对象是不能被改变的,内容改变就会产生新对象。那么StringBuffer就可以了,但只是改变其内容。不能改变外部变量所指向的内存地址)。 
b.传递地址值的数据类型:除String以外的所有复合数据类型,包括数组、类和接口 2 @+ p/ j) J7 C8 b7 {. |
; E! N3 }- x) C: ^
下面举例说明: 
在 Java 应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。 
Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。 
按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数,调用代码中的原始值也随之改变。 
当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。 
Java 应用程序按值传递所有参数,这样就制作所有参数的副本,而不管它们的类型。


   1. class Test  
       2. {  7 p; t$ S8 N" ~( d9 B# Z
       3.   public static void main(String args[])  
       4.   {  
       5.     int val;  
       6.     StringBuffer sb1, sb2;  
       7.    
       8.     val = 10;  # B4 P; V* Y. _) a) a& K
       9.     sb1 = new StringBuffer("apples");  
      10.     sb2 = new StringBuffer("pears");  
      11.     System.out.println("val is " + val);  6 f0 v$ N% d9 q3 j" e
      12.     System.out.println("sb1 is " + sb1);  
      13.     System.out.println("sb2 is " + sb2);  
      14.     System.out.println("");  6 E  ]* T6 K( r
      15.    
      16.     System.out.println("calling modify");  
      17.     //按值传递所有参数  7 T9 g" b) A) E. O" u
      18.     modify(val, sb1, sb2);  $ W8 p6 V) Y# {* E4 T# a: K  a0 n
      19.     System.out.println("returned from modify");  
      20.     System.out.println("");  + `6 p5 d0 ^% ~1 i
      21.    ! L1 `# N9 Z2 f2 ?# ]& M
      22.     System.out.println("val is " + val);  
      23.     System.out.println("sb1 is " + sb1);  , ^" y" e* K: c% J# A& C; Y
      24.     System.out.println("sb2 is " + sb2);  
      25.   }  
      26.      C" H9 Z- I2 ~3 _; ~5 Y
      27.   public static void modify(int a, StringBuffer r1,  2 J" i; L- J% G) |( ^; o# J9 L
      28.                             StringBuffer r2)  
      29.   {  : b3 c; ]. _" k6 a, J2 l
      30.       System.out.println("in modify...");  
      31.       a = 0;  
      32.       r1 = null;  //1  $ n3 A8 [5 l$ E7 S6 k( C" \( e/ `0 G
      33.       r2.append(" taste good");  : i; A/ B( `: r/ c% f3 [
      34.       System.out.println("a is " + a);  
      35.       System.out.println("r1 is " + r1);  
      36.       System.out.println("r2 is " + r2);  ) }( w! g' @8 Y6 S9 V5 ?) n
      37.   }  ; j  y* ?6 }0 Q) p/ u
      38. }  
      39.    
      - O! n5 b/ e$ E" c1 L% ]& r  }2 c
    Java 应用程序的输出  ) t! \2 B2 x/ K* y+ E, S. P. y( X
      
       1.    
       2. val is 10  
       3. sb1 is apples  ' ?9 ?, `2 u9 U7 h* w) d
       4. sb2 is pears  ' G+ S1 b0 Q$ t2 r, O
       5.    
       6. calling modify  0 ]/ |& G% ~+ X, \# I
       7. in modify...  
       8. a is 0  
       9. r1 is null  
      10. r2 is pears taste good  
      11. returned from modify  
      12.    
      13. val is 10  ' q$ s8 m" g  x6 s
      14. sb1 is apples  
      15. sb2 is pears taste good    z; Y2 ~+ N7 t8 Q2 L# R
      16.    
$ j9 a1 v5 x9 @3 W* n
& j# q) K. J# P: j& N% e
$ Z7 R# e4 ~4 M0 v  R

这段代码声明了三个变量:一个整型变量和两个对象引用。设置了每个变量的初始值并将它们打印出来。然后将所有三个变量作为参数传递给 modify 方法。 
- Z6 ~" B, [% F2 _% Z+ e
modify 方法更改了所有三个参数的值: 
* k7 W9 m1 S! T: a
将第一个参数(整数)设置为 0。 8 o( S" H# b" l1 t6 {; K4 M
将第一个对象引用 r1 设置为 null。 
保留第二个引用 r2 的值,但通过调用 append 方法更改它所引用的对象(这与前面的 C++ 示例中对指针 p 的处理类似)。 
7 q) A/ }1 w3 U2 k6 H/ g- R, g; ]
当执行返回到 main 时,再次打印出这三个参数的值。正如预期的那样,整型的 val 没有改变。对象引用 sb1 也没有改变。如果 sb1 是按引用传递的,正如许多人声称的那样,它将为 null。但是,因为 Java 编程语言按值传递所有参数,所以是将 sb1 的引用的一个副本传递给了 modify 方法。当 modify 方法在 //1 位置将 r1 设置为 null 时,它只是对 sb1 的引用的一个副本进行了该操作,而不是像 C++ 中那样对原始值进行操作。 4 \! v* e/ N* o8 j2 Y* D
$ s% d. p' p9 r+ K8 F6 C
另外请注意,第二个对象引用 sb2 打印出的是在 modify 方法中设置的新字符串。即使 modify 中的变量 r2 只是引用 sb2 的一个副本,但它们指向同一个对象。因此,对复制的引用所调用的方法更改的是同一个对象。 * A1 w0 n- z4 Z" c) `" _
传值---传递基本数据类型参数 
Java代码  收藏代码 ) N) o3 A: _. ?& Q- \
& M+ q( g0 d% R! t
    public    class           PassValue{  % I, e# x* S# Y6 y* _
        static void exchange(int a, int b){//静态方法,交换a,b的值  
            int temp;  
            temp = a;  
            a = b;  
            b = temp;  
        }  4 ]: T4 e8 v. |& j
        public static void main(String[] args){  
           int i = 10;  $ n7 _3 V1 B& Q; `+ u
           int j = 100;  : V' l6 x9 C8 w' {, ^
           System.out.println("before call: " + "i=" + i + "\t" + "j = " + j);//调用前  
            exchange(i, j);                                                                    //值传递,main方法只能调用静态方法  " h6 R* U* h8 Z
            System.out.println("after call: " + "i=" + i + "\t" + "j = " + j);//调用后  
        }  
    }  
    运行结果:  ( u; k, M+ E, v- {
            before call: i = 10        j = 100  
            after    call: i = 10        j = 100  3 g- v4 F5 D( O  O4 C
) v6 u* U9 l# X$ C. ^9 a( m. h

说明:调用exchange(i, j)时,实际参数i,j分别把值传递给相应的形式参数a,b,在执行方法exchange()时,形式参数a,b的值的改变不影响实际参数i和j的值,i和j的值在调用前后并没改变。 ( i1 H. h) r4 D8 j- S, o
引用传递---对象作为参数 8 f& Q, T* S* [! N; L# T% H
如果在方法中把对象(或数组)作为参数,方法调用时,参数传递的是对象的引用(地址),即在方法调用时,实际参数把对对象的引用(地址)传递给形式参数。这是实际参数与形式参数指向同一个地址,即同一个对象(数组),方法执行时,对形式参数的改变实际上就是对实际参数的改变,这个结果在调用结束后被保留了下来。 
Java代码  收藏代码 2 I, X& L8 e3 I" W1 @- j7 d/ k+ R

    class Book{  + W4 {8 i2 V( q6 G. ^4 v- d
        String name;  8 I8 g! l& m: F# d6 N+ r
        private folat price;  
        Book(String n,    float ){                //构造方法  1 q9 k9 k' `) d- q: ?
            name = n;  % y! a& v0 A& g0 Z  c
            price = p;  
        }  
        static  void  change(Book a_book,    String n,    float p){    //静态方法,对象作为参数  0 E( I6 Q- ^- d. k) ^0 x% \
                a_book.name = n;  
                a_book.price = p;  - _# X$ u  V' J
        }  ' T/ t4 D# Q6 K/ L% a7 F7 c3 q& d! \
        public void output(){        //实例方法,输出对象信息  + }7 M0 ?2 R) e  G$ d/ o' f
            System.out.println("name: " + name + "\t" + "price: " + price);  
        }  
    }  - O5 c9 S/ R, X+ C. I' K* m
     public class PassAddr{  
        public static void main(String [] args){  
            Book b = new Book("java2",    32.5f);  
            System.out.print("before call:\t");        //调用前  
            b.output();  * l7 |% M. e4 v+ ?+ T
            b.change(b,    "c++",    45.5f);            //引用传递,传递对象b的引用,修改对象b的值  : z2 N: H2 P: |1 ]) u
            System.out.print("after call:\t");            //调用后  + l6 k9 T0 e3 t* Y# q& M
            b.output();  : }, [) b* P# K2 x, D: d
        }  ( @( X3 d( g/ a0 r
    }  
    运行结果:  
            before    call:    name:java2        price:32.5  
            after       call:    name:c++          price:45.5  

# ~- @' g) O7 i5 g" O. A- ^7 t
说明:调用change(b,"c++",45.5f)时,对象b作为实际参数,把引用传递给相应的形式参数a_book,实际上a_book也指向同一个对象,即该对象有两个引用名:b和a_book。在执行方法change()时,对形式参数a_book操作就是对实际参数b的操作。
 

原文:http://tech.techweb.com.cn/thread-533188-1-1.html


原创粉丝点击