zoukankan      html  css  js  c++  java
  • java中的值传递

     如果参数类型是引用类型,那么传过来的就是这个引用参数的副本,这个副本存放的是参数的地址。如果在函数中没有改变这个副本的地址,而是改变了地址中的值,那么在函数内的改变会影响到传入的参数。如果在函数中改变了副本的地址,如new一个,那么副本就指向了一个新的地址,此时传入的参数还是指向原来的地址,所以不会改变参数的值

    ( 对象包括对象引用即地址和对象的内容)

     一句话就是:传入的对象参数,在方法中是改变对象的地址还是内容。

    面试题:当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?    答:是值传递。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代码 

    24 //传的是引用数据类型  

    25 public class Test {  

    26  

    27    public static void change(int[] counts) {  

    28        counts[0] = 6;  

    29        System.out.println(counts[0]);  

    30    }  

    31  

    32    public static void main(String[] args) {  

    33        int[] count = { 1, 2, 3, 4, 5 };  

    34        change(count);  

    35    }  

    36 }  

    37  

    38 在方法中 传递引用数据类型int数组,实际上传递的是其引用count的拷贝,他们都指向数组对象,在方法中可以改变数组对象的内容。即:对复制的引用所调用的方法更改的是同一个对象。  



    例3:

    Java代码 

    39 //对象的引用(不是引用的副本)是永远不会改变的  

    40 class A {  

    41    int i = 0;  

    42 }  

    43  

    44  

    45 public class Test {  

    46  

    47    public static void add(A a) {  

    48        a = new A();  

    49        a.i++;  

    50    }  

    51      

    52    public static void main(String args[]) {  

    53        A a = new A();  

    54        add(a);  

    55        System.out.println(a.i);  

    56    }  

    57 }  

    58  

    59 输出结果是0  

    60 在该程序中,对象的引用指向的是A ,而在change方法中,传递的引用的一份副本则指向了一个新的OBJECT,并对其进行操作。  

    61 而原来的A对象并没有发生任何变化。 引用指向的是还是原来的A对象。  




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

    Java代码 

    62  

    63 public class Example {  

    64    String str = new String("good");  

    65  

    66    char[] ch = { 'a', 'b', 'c' };  

    67  

    68    public static void main(String args[]) {  

    69        Example ex = new Example();  

    70        ex.change(ex.str, ex.ch);  

    71        System.out.print(ex.str + " and ");  

    72        System.out.println(ex.ch);  

    73    }  

    74  

    75    public void change(String str, char ch[]) {  

    76        str = "test ok";  

    77        ch[0] = 'g';  

    78    }  

    79 }   

    80 程序3输出的是 good and gbc.  

    81 String 比较特别,看过String 代码的都知道, String 是 final的。所以值是不变的。 函数中String对象引用的副本指向了另外一个新String对象,而数组对象引用的副本没有改变,而是改变对象中数据的内容.  

    82 对于对象类型,也就是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;

    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 应用程序的输出

    40 

    41 val is 10

    42 sb1 is apples

    43 sb2 is pears

    44 

    45 calling modify

    46 in modify...

    47 a is 0

    48 r1 is null

    49 r2 is pears taste good

    50 returned from modify

    51 

    52 val is 10

    53 sb1 is apples

    54 sb2 is pears taste good

    55 

    这段代码声明了三个变量:一个整型变量和两个对象引用。设置了每个变量的初始值并将它们打印出来。然后将所有三个变量作为参数传递给 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 + " " + "j = " + j);//调用前
    exchange(i, j); //值传递,main方法只能调用静态方法
    System.out.println("after call: " + "i=" + i + " " + "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 + " " + "price: " + price);
    }
    }
    public class PassAddr{
    public static void main(String [] args){
    Book b = new Book("java2", 32.5f);
    System.out.print("before call: "); //调用前
    b.output();
    b.change(b, "c++", 45.5f); //引用传递,传递对象b的引用,修改对象b的值
    System.out.print("after call: "); //调用后
    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的操作

  • 相关阅读:
    vue-cli项目中使用vw——相比flexible更原生的移动端解决方案
    android shap画圆(空心圆、实心圆)
    Android四大组件——Activity跳转动画、淡出淡入、滑出滑入、自定义退出进入
    HDU 3980 Paint Chain (sg函数)
    HDU 3951 Coin Game (简单博弈)
    HDU 1850 Being a Good Boy in Spring Festival (Nim博弈)
    HDU 3389 Game (阶梯博弈)
    HDU 3032 Nim or not Nim? (sg函数)
    HDU 1907 John (Nim博弈)
    HDU 4638 Group (线段树 | 树状数组 + 离线处理)
  • 原文地址:https://www.cnblogs.com/dengjiali/p/3492605.html
Copyright © 2011-2022 走看看