【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 中只有值传递。
相关推荐
向前看-2 小时前
验证码机制
前端·后端
xlsw_2 小时前
java全栈day20--Web后端实战(Mybatis基础2)
java·开发语言·mybatis
神仙别闹3 小时前
基于java的改良版超级玛丽小游戏
java
黄油饼卷咖喱鸡就味增汤拌孜然羊肉炒饭3 小时前
SpringBoot如何实现缓存预热?
java·spring boot·spring·缓存·程序员
暮湫3 小时前
泛型(2)
java
超爱吃士力架4 小时前
邀请逻辑
java·linux·后端
南宫生4 小时前
力扣-图论-17【算法学习day.67】
java·学习·算法·leetcode·图论
转码的小石4 小时前
12/21java基础
java
李小白664 小时前
Spring MVC(上)
java·spring·mvc
GoodStudyAndDayDayUp4 小时前
IDEA能够从mapper跳转到xml的插件
xml·java·intellij-idea