java是值传递还是引用传递

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 Tair(兼容Redis),内存型 2GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
简介: 本文澄清了Java中参数传递的常见误解,总结出Java采用“值传递”的方式。对于基本类型,传递其值的拷贝,方法内修改不影响原值;而对于对象类型,则传递其引用地址的拷贝,尽管是拷贝,但因指向同一对象,故方法内的修改会影响原对象状态。形参仅在方法内部有效,而实参则是调用方法时传递的具体值。通过示例和比喻(如复刻仓库钥匙),形象地解释了值传递、引用传递及Java特有的“共享对象传递”概念,帮助理解不同情况下参数传递的行为差异。

1.概述

曾经纠结了很久java的参数传递方式是什么样的,后面粗略的了解了一鳞半爪以后有了大概的印象:“传参数就是值传递,传对象就是引用传递”,后面进一步查找了相关资料和文章以后,发现这么理解是不正确的。

这里先放结论:

  • java中参数的传递可以理解为都是值传递
  • 基础数据类型传递的是值的拷贝
  • 对象类型是共享对象传递,传递的是地址的拷贝

2.形参和实参

要理解参数的传递就必须先理解形参和实参:

  • 形参:就是形式参数,用于定义方法的时候使用的参数,是用来接收调用者传递的参数的。
    形参只有在方法被调用的时候,虚拟机才会分配内存单元,在方法调用结束之后便会释放所分配的内存单元。
    因此,形参只在方法内部有效,所以针对引用对象的改动也无法影响到方法外。
  • 实参:就是实际参数,用于调用时传递给方法的参数

举个例子:

java

代码解读

复制代码

public static void main( String[] args ) {
    String string = "Hello";
    //string是实际参数
    sout(string);
}

public static void sout(String str){
    //str为形式参数
    System.out.println(str);
}

3.值传递和引用传递与共享对象传递

3.1.值传递和引用传递

理解了实参和形参,以及java对应的数据类型,我们就可以理解值传递和引用传递了。

  • 值传递:方法调用时,实际参数的被传递给对应的形式参数,函数接收的是原始值的一个copy, 此时内存中存在两个相等的基本类型,即实际参数和形式参数,后面方法中的操作都是对形参这个值的修改,不影响实际参数的值
  • 引用传递/址传递:方法调用时,实际参数的地址被传递给方法中相对应的形式参数,函数接收的是原始值的内存地址。在方法执行中,形参和实参内容相同,指向同一块内存地址,方法执行中对引用的操作将会影响到实际对象

对于这两种方式,网上有一个非常形象的图:

3.2.共享对象传递

但是java的传值策略有点类似于两者的结合,是共享对象传递

  • 共享对象传递:先获取到实际参数的地址,然后将其复制,并把该地址的拷贝传递给被调函数的形式参数。因为参数的地址都指向同一个对象,所以我们称也之为"传共享对象",所以,如果在被调函数中改变了形式参数的值,调用者是可以看到这种变化的。

这也是之所以说java也是值传递的原因,共享对象传递实际上也是对实参进行拷贝然后赋给形参,但是操作针对的对象不是值而是地址

由于传递的是地址的拷贝,所以如果你在方法中将这个地址指向了新的对象,实际上是没有任何对方法外是没有任何作用的,举个例子:

java

代码解读

复制代码

public static void main( String[] args ) {
    Person p = new Person();
    System.out.println("main中:" + p.hashCode());
    change(p);
    System.out.println("main中:" + p.hashCode());
}

public static void change(Person person){
    person = new Person();
    System.out.println("change中:" + person.hashCode());
}

//输出
main中:692404036
change中:1554874502
main中:692404036

可以看到在main方法中输出的hashCode指向的都是同一个对象,而change中指向了另一个,可以这么理解:

  • p为指向了第一个Person对象的地址
  • 把p拷贝了一份得到p‘,这里的p’就是change方法中的形参p
  • change中p指向了一个新的Person对象,在change这个函数范围里p指向的就是new出来的第二个Person对象的地址
  • 由于change中的p实际上是main中p的拷贝p‘,所以在change里p'指向的改变对main中的p不会有任何影响

总结

你在福建有座仓库,给自己配了一把钥匙

三种传递

  • 值传递:你建了一座一模一样的仓库给别人;
  • 引用传递:把你家仓库的钥匙给了别人;
  • 共享对象传递:把你家仓库钥匙复刻了一把给别人;

共享对象传递的特点

  • 拷贝的地址与原地址指向同一个内存对象:别人用你复刻的钥匙一样能进出你的仓库
  • 拷贝地址引用对象的改变不影响原地址的引用对象:老王在福建泉州也盖了个一模一样仓库,钥匙和你的一模一样,虽然你的仓库比较有名,但是唯独在泉州提到仓库大家都想到的是老王的仓库。你说这把钥匙能开仓库,在泉州大家想到是这把钥匙能开老王的仓库,在其他城市大家想到是能开你的仓库


转载来源:https://juejin.cn/post/7369470562143518730

相关文章
|
存储 Java 程序员
Java中相等比较与值传递
Java中相等比较与值传递
79 0
|
4月前
|
Java
java中的值传递和引用传递
【8月更文挑战第3天】在Java中,值传递用于基本数据类型,传递的是值的副本,因此方法内的修改不影响原值;而引用传递用于对象和数组,虽传递的是引用的副本,但对对象内容的修改会影响原始对象。理解这两者对于正确处理方法调用及参数至关重要。
|
4月前
|
Java
java中的值传递和引用传递
【8月更文挑战第2天】在Java中,基本数据类型如`int`、`double`等采用值传递,传递的是变量值的副本,因此方法内的修改不影响原变量。对象类型则通过引用传递,传递的是对象引用的副本,允许方法内修改原对象。例如,对`StringBuilder`对象的修改会影响原始对象。
|
7月前
|
存储 安全 Java
Java方法的值传递技术详解
Java方法的值传递技术详解
44 3
|
6月前
|
Java
Java的值传递与“引用传递”辨析
Java的值传递与“引用传递”辨析
30 0
|
7月前
|
JavaScript 前端开发 Java
【JAVA面试题】什么是引用传递?什么是值传递?
【JAVA面试题】什么是引用传递?什么是值传递?
|
7月前
|
Java
每日一道Java面试题:Java是值传递还是引用传递?
每日一道Java面试题:Java是值传递还是引用传递?
41 1
|
7月前
|
存储 Java
如何理解Java是按值传递
如何理解Java是按值传递