【Java基础】面试官:Java 对象是值传递还是引用传递?

前言

  • Java 值传递和引用传递一直值讨论比较多的话题,本文将结合概念和案例做一个比较详细的介绍。

形参和实参

  • 我们先了解一点前置知识,形参和实参,先说概念:形参出现在函数定义中,在整个函数体内都可使用,离开函数体则不可使用。实参出现在主调函数中,进入被调函数后,不能使用。
Java 复制代码
// 案例
public class MainTest {
    public static void main(String[] args) {
        // 这里的 a 就是实参
        int a = 1;
        func(a);
    }

    // param 则是形参
    public static void func(int param) {
        System.out.println(param);
    }
}

回顾一下 Java 内存管理

  • 我们先回顾一下这张熟悉的 Java 内存管理图片,在后文中我们会使用到。

值传递和引用传递

  • 我们先了解一下值传递和引用传递的概念:值传递是指在调用方式时,将实参的值拷贝一份给形参,对形参的修改不影响实参。引用传递也叫地址传递,指在调用方法时将实参的地址传递给形参,对形参的修改将影响实参的值,即传递的是实参的内存地址。

普通变量传递为值传递

  • 普通变量为值传递,这个大家应该比较好理解,其实就是拷贝了一份实参的值到形参中。
Java 复制代码
// 继续上面的案例
public class MainTest {
    public static void main(String[] args) {
        // 这里的 a 就是实参
        int a = 1;
        func(a);
        System.out.println("修改后实参的值:" + a);
    }

    // param 则是形参
    public static void func(int param) {
        System.out.println("形参的值:" + param);
        param++;
        System.out.println("修改后形参的值:" + param);
    }
}

// 输出
形参的值:1
修改后形参的值:2
修改后实参的值:1
  • 可以看到实参的值并未因为形参的修改而改变,即 Java 的普通变量传递为值传递。实际上是将实参的值拷贝了一份到栈帧的局部变量表中。

对象类型的传递是引用传递?

  • 比较有争议的是对象类型的传递,有的人认为是值传递,有的人认为是引用传递,我们先来看看两个案例:
Java 复制代码
// 案例一
public class MainTest {
    public static void main(String[] args) {
        TestObj testObj = new TestObj(0);
        System.out.println("原始实参的值:" + testObj.a);
        func(testObj);
        System.out.println("修改后实参的值:" + testObj.a);
    }

    public static void func(TestObj obj) {
        System.out.println("形参的值:" + obj.a);
        obj.a = 1;
        System.out.println("修改后形参的值:" + obj.a);
    }
}
class TestObj {
    int a;

    public TestObj(int a) {
        this.a = a;
    }
}

// 输出
原始实参的值:0
形参的值:0
修改后形参的值:1
修改后实参的值:1

// 对形参的修改影响了实参,难道是引用传递?那再看看下面的案例

// 案例二
public class MainTest {
    public static void main(String[] args) {
        TestObj testObj = new TestObj(0);
        System.out.println("原始实参的值:" + testObj.a);
        func(testObj);
        System.out.println("修改后实参的值:" + testObj.a);
    }

    public static void func(TestObj obj) {
        System.out.println("形参的值:" + obj.a);
        // 和案例一不同的改动
        obj=new TestObj(1);
        System.out.println("修改后形参的值:" + obj.a);
    }
}
class TestObj {
    int a;

    public TestObj(int a) {
        this.a = a;
    }
}

// 输出
原始实参的值:0
形参的值:0
修改后形参的值:1
修改后实参的值:0
  • 看到上面的两个案例,可能大家就比较疑惑了,几乎是两个相同的案例,都是对形参进行修改,一个却影响了实参,一个却没有影响实参,其实这就要从 Java 的内存管理说起。前面我们介绍了 Java 内存模型,我们看看两次调用形参和实参在 Java 内存中的分布。
  • 这样是不是很清晰了,Java 对象类型确实传递的是引用但只是引用的一个拷贝,可以理解为传递的是引用的拷贝值,对拷贝引用的修改并不会影响原引用,因此 Java 中只有值传递。
相关推荐
程序员黑豆26 分钟前
Java中怎么实现字符串拼接呢【AI全栈开发】
java
fox_lht43 分钟前
15.3.改进我们之前的输入、输出项目
开发语言·后端·学习·rust
大鸡腿同学1 小时前
用 AI 肝了一个星期的智能客服助手,看看怎么个事
后端
IT_陈寒1 小时前
Python的os.path.join居然能这么坑?
前端·人工智能·后端
java1234_小锋1 小时前
LangChain4j 开发Java Agent智能体- 多模态支持
java·开发语言·langchain4j
艳阳天_.1 小时前
星瀚弹框页面实现
java·前端·python
张忠琳1 小时前
【Go 1.26.4】Golang Channel 深度解析
开发语言·后端·golang
Rain5092 小时前
2.1 Nest.js 项目初始化与模块化架构
开发语言·前端·javascript·后端·架构·数据分析·node.js
cjp5602 小时前
009. ASP.NET WEB API 用户关联esp32设备
前端·后端·asp.net