【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 中只有值传递。
相关推荐
hgz07102 分钟前
Spring Boot Starter机制
java·spring boot·后端
daxiang120922053 分钟前
Spring boot服务启动报错 java.lang.StackOverflowError 原因分析
java·spring boot·后端
我家领养了个白胖胖4 分钟前
极简集成大模型!Spring AI Alibaba ChatClient 快速上手指南
java·后端·ai编程
jiayong234 分钟前
Markdown编辑完全指南
java·编辑器
heartbeat..22 分钟前
深入理解 Redisson:分布式锁原理、特性与生产级应用(Java 版)
java·分布式·线程·redisson·
一代明君Kevin学长25 分钟前
快速自定义一个带进度监控的文件资源类
java·前端·后端·python·文件上传·文件服务·文件流
aiopencode26 分钟前
上架 iOS 应用到底在做什么?从准备工作到上架的流程
后端
未来之窗软件服务27 分钟前
幽冥大陆(四十九)PHP打造Java的Jar实践——东方仙盟筑基期
java·php·jar·仙盟创梦ide·东方仙盟·东方仙盟sdk·东方仙盟一体化
普通网友33 分钟前
深入探讨Linux驱动开发:字符设备驱动开发与测试_linux 驱动开发设备号(2)
java·linux·驱动开发
4Forsee35 分钟前
【Android】动态操作 Window 的背后机制
android·java·前端