Java的参数传递是按值传递还是按引用传递?

Java的参数传递是按值传递还是按引用传递?

当一个对象被当作参数传递到一个方法后,在此方法内可以改变这个对象的属性,那么这里到底是按值传递还是按引用传递?

答: 是按引用值传递,非引用传递。
Java 语言的参数传递只有按值传递。 当一个实例对象作为参数被传递到方法中时,参数的值就是该对象的引用的一个副本。指向同一个对象,对象的内容可以在被调用的方法内改变,但对象的引用(不是引用的副本) 是永远不会改变的。

Java的参数传递,不管是基本数据类型还是引用类型的参数,都是按值传递,没有按引用传递!
这里写图片描述

1、基本数据类型的参数
先来看一下基本数据类型的参数按值传递的例子:

public class TransferTest {
    public static void main(String[] args) {
        int num = 1;
        System.out.println("changeNum()方法调用之前:num = " + num);
        changeNum(num);
        System.out.println("changeNum()方法调用之后:num = " + num);
    }

    public static void changeNum(int x) {
        x = 2;
         System.out.println(x);
    }
}

运行结果:
这里写图片描述

这个传递过程的示意图如下:
这里写图片描述

num作为参数传递给changeNum()方法时,是将内存空间中num所指向的那个存储单元中存放的值1传递给了changeNum()方法中的x变量,而这个x变量也在内存空间中分配了一个存储单元,这个时候,就把num的值1传递给了x的这个存储单元中。此后,在changeNum()方法中对x的一切操作都是针对x所指向的这个存储单元,与num所指向的那个存储单元没有关系了!

所以,在changeNum()方法调用之后,num所指向的存储单元的值还是没有发生变化,这就是“按值传递”! 按值传递的精髓是:传递的是存储单元中的内容,而不是存储单元的引用!

2、以引用类型作为参数传递:
同样,先看一个例子:

 public class Person {
    String name;
    public static void method(){};

}
public class TransferTest2 {


              public static void main(String[] args) {
                 Person person = new Person();
                 System.out.println(person);
                 change(person);

                 String name = person.name;
                 System.out.println(name);
                 System.out.println(person);
             }

             public static void change(Person p) {
                 System.out.println(p+"对person对象重设前");
                 p.name="Ted";
                 System.out.println(p+"对person对象重设后");

                  p = new Person();
                System.out.println(p);
            }
 }

运行结果:
这里写图片描述

这个传递过程的示意图如下:
这里写图片描述

可以看出经过cha nge()方法后person的地址值是一样的,即调用完change() 方法之后,person变量并没有发生改变。

程序在堆内存中开辟了一块内存空间用来存储Person类的实例对象,同时在栈内存中开辟了一个存储单元用来存储该实例对象的引用,即上图中person指向的存储单元。

当 person作为参数传递给change()方法,此时实际传入为person对象引用地址 。

在change()方法中,利用这个引用地址值,对person这个bean进行属性的重新设定,但重新设置后的对象,依然引用相同地址
这个说明一个道理
栈中有2个引用指向堆中的同一块地址(一个引用是在main方法中new 获得的person对象 另一个是change()方法中的形参p对象),且一个对象对其进行修改,另一个对象也将变化

ps(栈中怎么可以同时有多个Person类对象? 栈中运行机制为压栈弹栈,而此类中person 对象和 p对象一个属于main方法中,一个在change方法中,不冲突 )

当执行 p=new Person();时,在堆中重新开辟了一块内存,并将新的引用地址值反给栈的对象p.

转载至
https://github.com/nnngu/LearningNotes/blob/master/Java%20Basis/007%20Java%E7%9A%84%E5%8F%82%E6%95%B0%E4%BC%A0%E9%80%92%E6%98%AF%E5%80%BC%E4%BC%A0%E9%80%92%E8%BF%98%E6%98%AF%E5%BC%95%E7%94%A8%E4%BC%A0%E9%80%92.md

原文中有部分内容经过修改,告知。


猜你喜欢

转载自blog.csdn.net/ted_cs/article/details/82467216