Java 中的值传递

其实 Java 中是只存在值传递的,不存在引用传递。因为我们大多数人是从 C 语言入门,而 C 语言中是存在引用传递的,所以很容易在 Java 中混淆。

你还在为开发中频繁切换环境打包而烦恼吗?快来试试 Environment Switcher 吧!使用它可以在app运行时一键切换环境,而且还支持其他贴心小功能,有了它妈妈再也不用担心频繁环境切换了。https://github.com/CodeXiaoMai/EnvironmentSwitcher

数据类型分类

Java 中的数据类型分为两大类:基本类型对象类型,相对应的变量也分为两类:基本类型引用类型

// int 基本类型的数据类型,num 基本类型的变量,变量的值 10 直接保存在变量 num 中。
int num = 10;
// String 对象类型的数据类型,str 引用类型的变量,str 中保存的是 "hello" 在内存中的首地址。
String str = "hello";

下面这张图代表了变量中保存的值,以及实际对象在内存中的情况。
image

当修改变量的值时:

num = 20;
str = "java";

image

通过上图可以看出:
- 对于基本类型的变量 num,赋值运算符会直接改变它的值,原来的值被覆盖。
- 对于引用类型的变量 str,赋值运算符会将 str 中保存的内存地址(“hello”的地址)改为新的地址(“java”的地址),原来的地址被覆盖了,但是原来的地址指向的对象(“hello”)不会改变,只是没有任何引用指向它,会在垃圾回收机制触发时被回收

参数传递

这里首先要记住一点:参数传递就是赋值操作,也就是说,我们调用一个带参的函数时,形参和实参是两个变量,在内存中也是开辟了两个空间,只是把实参的值赋值给了形参。这是理解 Java 中只有值传递的关键。

下面就通过几个例子证明这一点。

例1.基本类型的变量传参

基本类型的变量传参

输出结果:

99

这个很好理解,其实通过编译器也可以发现,foo() 方法中的 num 颜色为灰色并且带有波浪线,把鼠标放到变量上还会看到提示:“Parameter can be converted to a local variable”。意思就是这个变量可以转化为一个本地变量,也就是在方法内部声明。根据“参数传递就是赋值操作”,来梳理一下整个过程。当调用 foo() 方法时,将实参( num) 的值赋值给了形参(num1 ),这时内存中存在两个变量 num(实参)、num1(形参),因为基本类型的变量赋值是直接覆盖操作,所以我们对形参(num1)的操作只是改变了形参的值,而不会影响实参(num) 。

例2.普通引用类型的变量传参

说是普通引用类型,是指类内部提供了改变自身的方法。

普通引用类型的变量传参

这次可以发现编译器没有提示异常信息,这至少可以证明,在 foo() 方法中对形参 myObject1 进行操作之后,形参 myObject1 仍然被引用,那是不是可以证明形参 myObject1 和实参 myObject 是同一个对象呢?我们先看看结果:

100

结果证明,foo 方法中的实参 myObject 和形参 myObject1 确实是指向同一个对象。再根据“参数传递就是赋值操作”这句话,来梳理一下。当调用 foo() 方法时,将实参(myObject) 的值(引用对象的内存地址)赋值给形参 (myObject1),这时内存中存在两个变量 myObject(实参)、myObject1(形参),但是它们都指向同一个内存地址,所以我们对形参(myObject1)的操作会影响实参(myObject) 。

例3.特殊引用类型的变量传参

上面提到了普通引用类型,当然就存在特殊引用类型了,它是指自身保存的值不可修改。如:String 和
Integer、Double、Boolean 等基本类型的包装类,它们都是 Immutable 类型的(它们的值都是 final 修饰的),所以每次对它们进行赋值操作,都是创建一个新的对象。

特殊引用类型的变量传参

我们发现 foo() 方法中的 value 被编译器提示可修改为本地变量。这就和例 1 中一样在 foo 方法中对形参的修改不会影响到实参,输出结果:

hello

这次根据“参数传递就是赋值操作”以及特殊引用类型的特点来梳理一下。当调用 foo() 方法时,将实参(value) 的值(引用对象的内存地址)赋值给形参 (value1),这时内存中存在两个变量 value(实参)、value1(形参),而且它们都指向同一个内存地址,但是当我们对形参(value1)进行赋值操作时,因为它是一个自身不可修改的特殊引用类型,所以”hello”对象并没有修改,而是在内存中又创建了一个值为”java”的新对象,并把“java”的地址赋值给形参,所以只是形参变了,而实参还是指向“hello”对象。

例4. 对普通引用类型使用赋值运算符

这次把 foo() 方法进行了修改,在方法内对形参进行重新赋值操作。

对普通引用类型使用赋值运算符

这时编译器又提示了形参 myObject 可以修改为本地变量。再根据“参数传递就是赋值操作”这句话,来梳理一下。当调用 foo() 方法时,将实参(myObject) 的值(引用对象的内存地址)赋值给形参 (myObject1),这时内存中存在两个变量 myObject(实参)、myObject1(形参),而且它们都指向同一个内存地址,但是在方法内部又对形参进行了一次赋值操作,这时形参指向了一个新的对象,而实参仍然指向原来的对象,这样形参和实参之间没有任何关联了。所以我们对形参(myObject1)的操作不会影响实参(myObject) 。

其实 foo() 方法等同于:

private static void foo() {
    MyObject myObject = new MyObject();
    myObject.num = 100;
}

同样可以看出 foo() 方法中的形参和实参完全没有关系了。

总结

Java 中参数传递其实就是赋值操作。
Java 中只存在值传递。

  • 对于基本类型变量,是把实参的值直接复制给形参。
  • 对于引用类型变量,是把实参引用的对象的内存地址复制给形参,所以实参和形参指向同一个对象。
  • 特殊引用类型变量,因为自身不可变的特点,当再次对形参进行赋值操作后,形参指向一个新的对象,而实参扔指向原来的对象。特殊引用类型变量包括 String 和一些基本类型的包装类(Integer、Double、Boolean等)。
<think>好的,用户想了解Java中的值传递原理和实现机制。首先,我需要确认自己对值传递的理解是否正确。Java中确实只有值传递,无论是基本类型还是对象引用。当传递基本类型时,传递的是值的副本;当传递对象时,传递的是对象引用的副本。这意味着在方法内部修改引用指向的对象会影响原对象,但修改引用本身不会影响原引用。 接下来,我需要结合用户提供的引用内容。用户引用了关于反射机制、值传递与引用传递的例子、形参传递机制以及TransmittableThreadLocal的内容。其中引用[2]和[3]直接相关,特别是引用[2]中的例子展示了基本类型的值传递,修改形参不影响实参。引用[3]提到值传递机制,可能与参数传递的底层实现有关。 用户可能存在的疑问是为什么对象作为参数传递时,修改对象内容会影响原对象,但重新赋值引用却不会。需要解释清楚传递的是引用的副本,副本和原引用指向同一个对象,所以修改对象属性会影响原对象,但修改副本引用指向新对象不会影响原引用。 另外,用户可能混淆了值传递和引用传递的概念,需要明确Java中没有引用传递,只有值传递,包括对象引用的传递也是值传递的一种形式。需要举例说明基本类型和对象类型的不同表现,并解释底层原理,如栈内存中的副本机制。 可能还需要提到内存模型,比如栈和堆的区别,基本类型存储在栈,对象实例在堆,引用变量在栈中存储地址。传递时复制的是栈中的值,即地址的副本,因此操作同一对象实例但引用变量本身是独立的。 最后,结合用户提供的引用,引用[2]的例子可以用于说明基本类型的值传递,而引用[3]中的形参传递机制可以进一步支持值传递的解释。可能需要强调无论参数类型如何,传递的都是值的副本,只是这个值可能是基本类型的数值或对象的引用地址。</think>### Java 值传递原理及实现机制 #### 一、基本概念 Java **严格采用值传递机制**,这意味着方法调用时传递的始终是变量的副本,而非变量本身。无论是基本数据类型还是对象引用类型,传递的都是值的拷贝[^2][^3]。 #### 二、实现原理 1. **基本数据类型传递** - 传递的是变量的**值副本**,方法内修改形参不影响原始变量。 - 例如: ```java int x = 10; void modify(int num) { num = 20; } // 调用 modify(x) 后,x 仍为 10 ``` 2. **对象引用类型传递** - 传递的是对象引用的**地址值副本**,形参和实参指向同一对象。 - 若通过形参修改对象属性,会影响原始对象;但若修改形参的引用指向,不影响原始引用。 - 示例: ```java class MyObj { int val; } MyObj obj = new MyObj(); void modifyRef(MyObj param) { param.val = 20; // 修改对象属性,原始对象受影响 param = new MyObj(); // 修改形参引用指向,原始引用不受影响 } ``` #### 三、内存模型解释 - **栈内存**:存储基本数据类型值和对象引用地址。 - **堆内存**:存储对象实例。 - 值传递过程中,栈内存中的值(基本类型数值或引用地址)会被复制到新的栈帧中。 #### 四、典型场景验证 1. **基本类型示例** 引用[^2]中的代码: ```java public static void changeValue(int num) { num = 20; } ``` 调用后原始变量不变,验证了值副本机制。 2. **对象类型示例** ```java void updateArray(char[] arr) { arr[0] = 'X'; // 修改堆内存中的对象 arr = new char[5]; // 修改栈中的引用副本 } ``` 执行后原始数组的第一个元素会被修改,但数组引用不变。 #### 五、与引用传递的对比 - **值传递**:传递副本,副本与原数据独立(但对象引用副本与原引用指向同一对象)。 - **引用传递**(Java不存在):直接操作原始变量,如C++的引用参数。 #### 六、特殊场景扩展 多线程环境下,父子线程间的值传递可通过`TransmittableThreadLocal`实现,其本质仍是基于值传递机制的特殊封装[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值