• Java到底是值传递还是引用传递?


    在 Java 语言中,本质只有值传递,而无引用传递,解释和证明详见正文。

    说到值传递和引用传递我们不得不提到两个概念:值类型引用类型

     

    一、值类型

     

    通俗意义上来说,所谓的值类型指的就是 Java 中的 8 大基础数据类型:

    • 整数型:byte、int、short、long
    • 浮点型:float、double
    • 字符类型:char
    • 布尔类型:boolean

     

     从 JVM 层面来讲:所谓的值类型指的是在赋值时,直接在栈中(Java 虚拟机栈)生成值的类型,如下图所示:

     

    二、引用类型

    引用类型是指除值类型之外的数据类型,比如:

    • 接口
    • 数组
    • 字符串
    • 包装类(Integer、Double...)

     从 JVM 的层面来讲,所谓的引用类型是指,在初始化时将引用生成栈上,而值生成在堆上的这些数据类型,如下图所示:

    三、值传递

    值传递(Pass By Value)指的是方法传参时,传递的是原内容的副本,因此对副本进行如何修改都不会影响原内容。

    实现代码如下:

    public class PassTest {
        public static void main(String[] args) {
            int age = 18;
            System.out.println("调用方法前:" + age);
            intTest(age);
            System.out.println("调用方法后:" + age);
        }
    
        private static void intTest(int age) {
            age = 30;
            System.out.println("方法中修改为:" + age);
        }
    }

    程序的执行结果为:

    调用方法前:18

    方法中修改为:30

    调用方法后:18

    从上述结果可以看出,在方法中修改参数并未影响原内容,我们把这种传参方式称之为值传递。

    四、引用传递

    引用传递(Pass By Reference)指的是方法传参时,传递的是参数本身,因此对参数进行任意修改都会影响原内容。

    模拟“引用传递”的实现代码如下:

    public class PassTest {
        public static void main(String[] args) {
            char[] name = {'磊', '哥'};
            System.out.println("调用方法前:" + new String(name));
            paramTest(name);
            System.out.println("调用方法后:" + new String(name));
        }
        private static void paramTest(char[] n) {
            n[1] = '神';
            System.out.println("方法中修改为:" + new String(n));
        }
    }

    程序的执行结果为:

    调用方法前:磊哥

    方法中修改为:磊神

    调用方法后:磊神

    从上述的结果可以看出在 paramTest 方法中修改了参数之后,在 main 方法中再打印参数时,发现参数的值也跟着发生了改变,那么似乎我们可以得出结论,Java 中貌似也有“引用传递”,然而实事并如此,我们接着看。

    五、真假“引用传递”

    我们给上面的代码添加一行,如下所示:

    public class PassByValue {
        public static void main(String[] args) {
            char[] name = {'磊', '哥'};
            System.out.println("调用方法前:" + new String(name));
            paramTest(name);
            System.out.println("调用方法后:" + new String(name));
        }
        private static void paramTest(char[] n) {
            n = new char[2]; // 添加此行代码
            n[1] = '神';
            System.out.println("方法中修改为:" + new String(n));
        }
    }

    程序的执行结果为:

    调用方法前:磊哥

    方法中修改为:神

    调用方法后:磊哥

    从上述结果可以看出,当我们在 paramTest 方法中添加 new char[] 之后,“引用传递”就突然变值传递了?为什么?

    这是因为,在 Java 语言中本质上只有值传递,也就说 Java 的传参只会传递它的副本,并不会传递参数本身

    前面那个带引号的“引用传递”其实只是传递了它的引用副本,如下图所示:

    PS:《Java虚拟机规范》中对  Java 堆的描述是:“所有的对象实例以及数组都应当在堆上分配”。

    所以我们在调用 new char[] 之后,可以看出 n 对象有了新地址,而原内容并未被修改,如果按照引用传递的思路来看的话,不管执行任何方式的修改都会改变原内容,因此我们可以更加确认 Java 语言中只有值传递,如下图所示:

     六、总结

    通过本文的内容,我们可以得出:在 Java 语言中只有值传递,方法传参时只会传递副本信息而非原内容。我们还知道了基础数据类型会直接生成到栈上,而对象或数组则会在栈和堆上都生成信息,并将栈上生成的引用,直接指向堆中生成的数据,如下图所示:

    原文出处:

    CSDN微信公众号, 多图证明,Java到底是值传递还是引用传递?https://mp.weixin.qq.com/s/hwRrRnayu8TNgoL2X1_EOA, 2020-09-03

     

     

     

     

  • 相关阅读:
    python处理csv数据
    python数据持久存储:pickle模块的基本使用
    使用SVD方法实现电影推荐系统
    使用矩阵分解(SVD)实现推荐系统
    多维数组分解----SVD在推荐系统中的应用-
    Logistic Regression--逻辑回归算法汇总**
    Netflix推荐系统:从评分预测到消费者法则
    从决策树学习谈到贝叶斯分类算法、EM、HMM
    数据挖掘中 决策树算法实现——Bash
    决策树算法
  • 原文地址:https://www.cnblogs.com/ryelqy/p/13607917.html
Copyright © 2020-2023  润新知