java值传递和引用传递

转载的,值传递和引用传递弄晕了。

Java中没有指针,所以也没有引用传递了,仅仅有值传递 不过可以通过对象的方式来实现引用传递 类似java没有多继承 但可以用多次implements 接口实现多继承的功能
值传递:方法调用时,实际参数把它的值传递给对应的形式参数,方法执行中形式参数值的改变不影响实际参 

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

Java参数按值传递

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

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

在 Java 应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。
  
Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。
   
按 值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味 着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数的值,调用代码中的原始值也随之改变。如果 函数修改了该参数的地址,调用代码中的原始值不会改变.
   
当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。 
   
Java 应用程序按值传递参数(引用类型或基本类型),其实都是传递他们的一份拷贝.而不是数据本身.(不是像 C++ 中那样对原始值进行操作。) 


例1:

Java代码  复制代码
  1. //在函数中传递基本数据类型,   
  2. public   class  Test {   
  3.             
  4.      public   static   void  change( int  i,  int  j) {   
  5.          int  temp = i;   
  6.         i = j;   
  7.         j = temp;   
  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. }   
  19.   
  20. 结果为:   
  21. a= 3   
  22. b= 4   
  23. 原因就是 参数中传递的是 基本类型 a 和 b 的拷贝,在函数中交换的也是那份拷贝的值 而不是数据本身;  

 

例2:

Java代码  复制代码
  1. //传的是引用数据类型   
  2. public   class  Test {   
  3.   
  4.      public   static   void  change( int [] counts) {   
  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);   
  12.     }   
  13. }   
  14.   
  15. 在方法中 传递引用数据类型 int 数组,实际上传递的是其引用count的拷贝,他们都指向数组对象,在方法中可以改变数组对象的内容。即:对复制的引用所调用的方法更改的是同一个对象。  


例3:
Java代码  复制代码
  1. //对象的引用(不是引用的副本)是永远不会改变的   
  2. class  A {   
  3.      int  i =  0 ;   
  4. }   
  5.   
  6.   
  7. public   class  Test {   
  8.   
  9.      public   static   void  add(A a) {   
  10.         a =  new  A();   
  11.         a.i++;   
  12.     }   
  13.        
  14.      public   static   void  main(String args[]) {   
  15.         A a =  new  A();   
  16.         add(a);   
  17.         System.out.println(a.i);   
  18.     }   
  19. }   
  20.   
  21. 输出结果是 0   
  22. 在该程序中,对象的引用指向的是A ,而在change方法中,传递的引用的一份副本则指向了一个新的OBJECT,并对其进行操作。   
  23. 而原来的A对象并没有发生任何变化。 引用指向的是还是原来的A对象。  




例4: 
String 不改变,数组改变

Java代码  复制代码
  1.   
  2. public   class  Example {   
  3.     String str =  new  String( "good" );   
  4.   
  5.      char [] ch = {  'a' ,  'b' ,  'c'  };   
  6.   
  7.      public   static   void  main(String args[]) {   
  8.         Example ex =  new  Example();   
  9.         ex.change(ex.str, ex.ch);   
  10.         System.out.print(ex.str +  " and " );   
  11.         System.out.println(ex.ch);   
  12.     }   
  13.   
  14.      public   void  change(String str,  char  ch[]) {   
  15.         str =  "test ok" ;   
  16.         ch[ 0 ] =  'g' ;   
  17.     }   
  18. }    
  19. 程序 3 输出的是 good and gbc.   
  20. String 比较特别,看过String 代码的都知道, String 是  final 的。所以值是不变的。 函数中String对象引用的副本指向了另外一个新String对象,而数组对象引用的副本没有改变,而是改变对象中数据的内容.   
  21. 对于对象类型,也就是Object的子类,如果你在方法中修改了它的成员的值,那个修改是生效的,方法调用结束后,它的成员是新的值,但是如果你把它指向一个其它的对象,方法调用结束后,原来对它的引用并没用指向新的对象。 
Java参数,不管是原始类型还是引用类型,传递的都是副本(有另外一种说法是传值,但是说传副本更好理解吧,传值通常是相对传址而言)。
    如果参数类型是原始类型,那么传过来的就是这个参数的一个副本,也就是这个原始参数的值,这个跟之前所谈的传值是一样的。如果在函数中改变了副本的 值不会改变原始的值.
    如果参数类型是引用类型,那么传过来的就是这个引用参数的副本,这个副本存放的是参数的地址。如果在函数中没有改变这个副本的地址,而是改变了地址中的 值,那么在函数内的改变会影响到传入的参数。如果在函数中改变了副本的地址,如new一个,那么副本就指向了一个新的地址,此时传入的参数还是指向原来的 地址,所以不会改变参数的值。
 ( 对象包括对象引用即地址和对象的内容)
 
a.传递值的数据类型:八种基本数据类型和String(这样理解可以,但是事实上String也是传递的地址,只是string对象和其他对 象是不同的,string对象是不能被改变的,内容改变就会产生新对象。那么StringBuffer就可以了,但只是改变其内容。不能改变外部变量所指 向的内存地址)。
b.传递地址值的数据类型:除String以外的所有复合数据类型,包括数组、类和接口

下面举例说明:
   Java   应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。  Java   应用程序按引用传递对象这一事实并不意味着   Java   应用程序按引用传递参数。参数可以是对象引用,而   Java   应用程序是按值传递对象引用的。
Java   应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。
按 值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味 着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数,调用代码中的原始值也随之改变。
当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在   C++   中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。   Java   应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。
Java   应用程序按值传递所有参数,这样就制作所有参数的副本,而不管它们的类型。
  1. class  Test
  2. {
  3.   public  static  void  main(String  args[])
  4.   {
  5.     int  val;
  6.     StringBuffer  sb1, sb2;
  7.  
  8.     val = 10;
  9.     sb1 = new  StringBuffer ("apples" );
  10.     sb2 = new  StringBuffer ("pears" );
  11.     System .out.println("val is "  + val);
  12.     System .out.println("sb1 is "  + sb1);
  13.     System .out.println("sb2 is "  + sb2);
  14.     System .out.println("" );
  15.  
  16.     System .out.println("calling modify" );
  17.     // 按值传递所有参数
  18.     modify(val, sb1, sb2);
  19.     System .out.println("returned from modify" );
  20.     System .out.println("" );
  21.  
  22.     System .out.println("val is "  + val);
  23.     System .out.println("sb1 is "  + sb1);
  24.     System .out.println("sb2 is "  + sb2);
  25.   }
  26.  
  27.   public  static  void  modify(int  a, StringBuffer  r1,
  28.                             StringBuffer  r2)
  29.   {
  30.       System .out.println("in modify..." );
  31.       a = 0;
  32.       r1 = null ;  //1
  33.       r2.append(" taste good" );
  34.       System .out.println("a is "  + a);
  35.       System .out.println("r1 is "  + r1);
  36.       System .out.println("r2 is "  + r2);
  37.   }
  38. }
  39.  
Java   应用程序的输出
  1.  
  2. val is 10
  3. sb1 is apples
  4. sb2 is pears
  5.  
  6. calling modify
  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
  14. sb1 is apples
  15. sb2 is pears taste good
  16.  


这段代码声明了三个变量:一个整型变量和两个对象引用。设置了每个变量的初始值并将它们打印出来。然后将所有三个变量作为参数传递给   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   的一个副本,但它们指向同一个对象。因此,对复制的引用所调用的方法更改的是同一个对象。
传值---传递基本数据类型参数 
public    class           PassValue{
    static void exchange(int a, int b){//静态方法,交换a,b的值
        int temp;
        temp = a;
        a = b;
        b = temp;
    }
    public static void main(String[] args){
       int i = 10;
       int j = 100;
       System.out.println("before call: " + "i=" + i + "/t" + "j = " + j);//调用前
        exchange(i, j);                                                                    //值传递,main方法只能调用静态方法
        System.out.println("after call: " + "i=" + i + "/t" + "j = " + j);//调用后
    }
}
运行结果:
        before call: i = 10        j = 100
        after    call: i = 10        j = 100
说明:调用exchange(i, j)时,实际参数i,j分别把值传递给相应的形式参数a,b,在执行方法exchange()时,形式参数a,b的值的改变不影响实际参数i和j的值,i和j的值在调用前后并没改变。
引用传递---对象作为参数
如果在方法中把对象(或数组)作为参数,方法调用时,参数传递的是对象的引用(地址),即在方法调用时,实际参数把对对象的 引用(地址)传递给形式参数。这是实际参数与形式参数指向同一个地址,即同一个对象(数组),方法执行时,对形式参数的改变实际上就是对实际参数的改变, 这个结果在调用结束后被保留了下来。
class Book{
    String name;
    private folat price;
    Book(String n,    float ){                //构造方法
        name = n;
        price = p;
    }
    static  void  change(Book a_book,    String n,    float p){    //静态方法,对象作为参数
            a_book.name = n;
            a_book.price = p;
    }
    public void output(){        //实例方法,输出对象信息
        System.out.println("name: " + name + "/t" + "price: " + price);
    }
}
 public class PassAddr{
    public static void main(String [] args){
        Book b = new Book("java2",    32.5f);
        System.out.print("before call:/t");        //调用前
        b.output();
        b.change(b,    "c++",    45.5f);            //引用传递,传递对象b的引用,修改对象b的值
        System.out.print("after call:/t");            //调用后
        b.output();
    }
}
运行结果:
        before    call:    name:java2        price:32.5
        after       call:    name:c++          price:45.5
说明:调用change(b,"c++",45.5f)时,对象b作为实际参数,把引用传递给相应的形式参数a_book,实际上a_book也指向同一 个对象,即该对象有两个引用名:b和a_book。在执行方法change()时,对形式参数a_book操作就是对实际参数b的操作。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值