写点什么

值传递 OR 引用传递?大部人都答错了!

作者:王磊
  • 2024-09-04
    陕西
  • 本文字数:1972 字

    阅读完需:约 6 分钟

Java 是值传递还是引用传递?这是 Java 中比较基础的一道常见面试题,但对于这道问题的大部分答案都是错的,大部人会这样回答这个问题:


在 Java 中,如果传递的是基本数据类型,那么就是值传递;而如果传递的是对象或数组的话,那么就是引用传递。


然而,这个答案是错的!

定义描述

值传递和引用传递是编程中参数传递给方法时的两种方式,它们的定义如下:


  1. 值传递(Pass by Value):在值传递中,实际参数的值被复制一份,然后将这份复制的值传递给函数或方法的相应参数。因此,函数或方法内对参数所做的任何修改都不会影响到实际参数的值。

  2. 引用传递(Pass by Reference):在引用传递中,传递给方法的是实际参数的引用(或地址)。这意味着方法内对参数所做的任何修改都会直接影响到实际参数。


需要注意的是,有些编程语言,如 C++ 提供了真正的引用传递机制,允许你直接传递变量的引用,并且可以在函数或方法中改变这个引用的指向。而在 Java 中,即使是对象,也是通过值传递的,只不过这个值是对象引用副本(而非对象引用本身)

正确结论

在 Java 中,(传递参数时)无论是基本数据类型还是对象(或数组),使用的都是值传递的方式。只是对于对象(或数组)而言,传递的值是对象引用副本,而非对象引用本身。


在 Java 中,只有值传递没有引用传递

举个例子

例如 Integer 是包装类对象吧?它不是基本数据类型对吧,当我们传递 Integer 对象时,在新方法所做的所有修改,并不会影响原对象本身,具体示例代码如下:


public class PassExample {    public static void main(String[] args) {        Integer number = new Integer(10);        method(number);        System.out.println("number:" + number); // 输出:number:10    }    public static void method(Integer number) {        number = 20; // 修改 num 的值,不会影响原始变量的值    }}
复制代码


以上程序的执行结果如下:



从上述结果可以看出,当传递的是 Integer 对象时,其依然是值传递,所以在 Java 语言中,并没有引用传递。


因此,无论是基础数据类型,还是引用数据类型(对象),都为值传递,而非引用传递。

特殊的例子

有人说:不对啊,磊哥,你看我传递数组时,改变传递的数组就会影响原数组啊,具体示例如下:


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


以上程序的执行结果为:


调用方法前:磊哥

方法中修改为:磊神

调用方法后:磊神


这样就出问题了,当传递了数组之后,明显是“引用传递”,而非值传递,这到底是怎么回事?


别着急,当我们把新方法中的代码做了以下调整之后,运行结果又不一样了,如下代码所示:


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


以上程序的执行结果为:


调用方法前:磊哥

方法中修改为: 神

调用方法后:磊哥


你会发现,当我们在新方法中仅仅添加了一行“n = new char[2];”代码时,它又变成了值传递,这是怎么回事?

原因分析

如果是引用传递,那么我在新方法中无论如何修改,那么都应该是影响原对象才对,而刚才我稍微调整了代码之后就发现其并非引用传递,而是值传递,这是因为当传递数组时,其传递的是“引用副本”,而非真正的引用对象(也就是其本身)


也就说,当传递数组时,其实传递的是“引用副本”,如下图所示:



然而,在调用了“n = new char[2];”代码之后,给变量在堆上创建了新对象,此时就不再使用原来的引用副本了,这个时候,再修改新方法中的变量就不影响原变量了,如下图所示:



所以,在 Java 中,只有值传递,它始终传递的都是副本,而非原(引用)对象

小结

在 Java 中,(传递参数时)无论是基本数据类型还是对象(或数组),使用的都是值传递的方式。只是对于对象(或数组)而言,传递的值是对象引用副本,而非对象引用本身。


本文已收录到我的面试小站 www.javacn.site,其中包含的内容有:Redis、JVM、并发、并发、MySQL、Spring、Spring MVC、Spring Boot、Spring Cloud、MyBatis、设计模式、消息队列等模块。

用户头像

王磊

关注

javacn.site 2018-08-25 加入

我的小站:javacn.site

评论

发布
暂无评论
值传递OR引用传递?大部人都答错了!_王磊_InfoQ写作社区