Java到底是引用传递还是值传递

Wesley13
• 阅读 1172

前言

前段时间在群里看到类似这样一个问题,下面的代码会输出什么呢?

public void test() {    String str = "hello";    change(str);    System.out.println(str); } private void change(String str) {    str = "world"; }

当时看到这题,瞬间勾起了我的回忆。遥想当年,也曾经碰到过类似的问题,当时研究了好久才搞明白,这里再记录一下这个问题的思路。

先来说一下答案:输出:hello

解决这类问题首先要搞明白Java到底是引用传递还是值传递

Java到底是引用传递还是值传递

首先来解释一下什么是引用传递,什么是值传递。

  • 引用传递(pass by reference)是指在调用方法时将实际参数的地址直接传递到方法中,那么在方法中对参数所进行的修改,将影响到实际参数。

  • 值传递(pass by value)是指在调用方法时将实际参数拷贝一份传递到方法中,这样在方法中如果对参数进行修改,将不会影响到实际参数。

那在Java中到底是引用传递还是值传递呢?其实这个问题也一直是争论不断,而且官方也没给个确切答案。但是就我个人理解,Java是值转递

我们先来看一个简单的例子:

public void test() {     int a = 1;     change(a);     System.out.println("a的值:" + a); } private void change(int a) {     a = a + 1; } // 输出 a的值:1

在test()方法中定义了一个基本类型的变量a,然后调用change()方法试图改变这个变量,最后输出的还是原来的值。

首先我们要清楚,一个方法中的局部变量是存在栈中的,如果是基本类型的变量则直接存的是这个变量的值,如果是引用类型的变量则存的是值的地址,指向堆中具体的值。

上面的例子中,调用change()方法传递的a,其实是a变量的拷贝,不是真正的a,在change()方法中改变的是拷贝,对真正的a是没有影响的。

这么一看,Java确实是值传递,但是我们再看下面这个例子,你就会纠结了

public void test() {    User user = new User();    user.setAge(18);    change(user);    System.out.println("年龄:" + user.getAge()); } private void change(User user) {    user.setAge(19); } // 输出 年龄:19

看,对象里的属性被改变了,不是值传递吗,应该不会改变啊,这时候就有人总结了,当传的值是基本类型时是值传递、当传的是引用类型时是引用传递。真的是这样吗?

分析这个问题,我们需要知道变量在jvm中是怎么存储的。

首先看基本类型,这个很简单,变量在栈中直接存的是值,传到change()方法的是这个变量的拷贝,因此对拷贝的变量修改不会影响原变量的值。

接着看引用类型,变量在栈中存储的是引用地址,这个地址指向堆中具体的值,如下图:

Java到底是引用传递还是值传递

当调用change()方法传入变量时,也是拷贝变量,但是这里的拷贝只是栈中的引用地址,并不会拷贝堆中的数据,因此会变成下图这样:

Java到底是引用传递还是值传递

虽然变量是拷贝,但是指向的地址是同一个,因此对变量中的数据修改时,还是会影响到原来真实的变量,但是,如果我们修改的是变量在栈中的地址,则不会影响原变量,例如下面这段代码:

public void test() {    User user = new User();    user.setAge(18);    change(user);    System.out.println("年龄:" + user.getAge()); } private void change(User user) {    user = new User();    user.setAge(19); } // 输出 年龄:18

这种是修改变量在栈中的地址,则不会影响原变量。

说到这里,大家差不多懂了,但是回头看最开始的那个问题,传入String类型的变量,String是引用类型,按道理,原变量是会被改变的呀,结果怎么是不变呢?

String变量比较特殊,我们看String的源码可以知道,String的值是通过内部的char[]数组来维护的,但是这个数据定义的是final类型的,因此,String的值是不可变的。我们平时修改String的值,其实是重新new了一个String对象,例如下面这段代码:

String a = "hello"; a = "world";

这段代码里,其实a变量并没有被修改成world,只是重新new了一个String对象,这个对象的值是world,并把这个对象的引用地址赋给了a,原来的hello还是在堆中,只是这个值没有被引用,过段时间会被gc垃圾回收。

String变量传值在内存中的变化如下图:

Java到底是引用传递还是值传递

String拷贝的是变量地址,但是它改变不了原String的值,因为String是不可变的,所以在change()方法中是重新new了一个String对象,改变的是新对象的值,原变量是没有影响的。

结论

Java是值传递当传的是基本类型时,传的是值的拷贝,对拷贝变量的修改不影响原变量;当传的是引用类型时,传的是引用地址的拷贝,但是拷贝的地址和真实地址指向的都是同一个真实数据,因此可以修改原变量中的值;当传的是String类型时,虽然拷贝的也是引用地址,指向的是同一个数据,但是String的值不能被修改,因此无法修改原变量中的值。


扫一扫,关注我

Java到底是引用传递还是值传递

[

数据库索引

2020-08-02

Java到底是引用传递还是值传递

](https://www.oschina.net/action/GoToLink?url=http%3A%2F%2Fmp.weixin.qq.com%2Fs%3F__biz%3DMzU2MDQ0NTI0OQ%3D%3D%26mid%3D2247484333%26idx%3D1%26sn%3D8313747b10396e2e7d34783634067372%26chksm%3Dfc06a183cb712895a55e7e3b1f70635147c7ce3be65eb35da00924188629b2fdf8f6934c42a1%26scene%3D21%23wechat_redirect)

[

事务:不好意思,你被隔离了!

2020-07-23

Java到底是引用传递还是值传递

](https://www.oschina.net/action/GoToLink?url=http%3A%2F%2Fmp.weixin.qq.com%2Fs%3F__biz%3DMzU2MDQ0NTI0OQ%3D%3D%26mid%3D2247484223%26idx%3D1%26sn%3D15776b9b6f43a26d9750a4baa089e16c%26chksm%3Dfc06a111cb71280726003c8911d9bf53a392a2ac4486a0dc7ed94d681070c3a980c794220392%26scene%3D21%23wechat_redirect)

[

spring事务咋和新冠病毒一样,还会传染?

2020-07-05

Java到底是引用传递还是值传递

](https://www.oschina.net/action/GoToLink?url=http%3A%2F%2Fmp.weixin.qq.com%2Fs%3F__biz%3DMzU2MDQ0NTI0OQ%3D%3D%26mid%3D2247484180%26idx%3D1%26sn%3Dd053e5901940a18b16708edd49f8aef6%26chksm%3Dfc06a13acb71282c9f4df086b5ef34be3bc549ad5d44919802865d05d9eec5819a55590388cb%26scene%3D21%23wechat_redirect)

[

数据是怎么一步一步到服务器的

2020-06-18

Java到底是引用传递还是值传递

](https://www.oschina.net/action/GoToLink?url=http%3A%2F%2Fmp.weixin.qq.com%2Fs%3F__biz%3DMzU2MDQ0NTI0OQ%3D%3D%26mid%3D2247484154%26idx%3D1%26sn%3D5aae52994bdb508dbce52f339ba80967%26chksm%3Dfc06a0d4cb7129c2208edd849259e09ef9979d776e5243edf9250bc5b8fb693411deb18efb3c%26scene%3D21%23wechat_redirect)

[

redis分布式锁

2020-06-05

Java到底是引用传递还是值传递

](https://www.oschina.net/action/GoToLink?url=http%3A%2F%2Fmp.weixin.qq.com%2Fs%3F__biz%3DMzU2MDQ0NTI0OQ%3D%3D%26mid%3D2247484043%26idx%3D1%26sn%3Dfb04445f7128403539153731b71f95d7%26chksm%3Dfc06a0a5cb7129b3efd8ef4fe2b482439615d687bbc25f410b3d2bd733b710dd550660bd0c9e%26scene%3D21%23wechat_redirect)

本文分享自微信公众号 - pipi蛋(pipidan_fuyun)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
5个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Stella981 Stella981
3年前
HIVE 时间操作函数
日期函数UNIX时间戳转日期函数: from\_unixtime语法:   from\_unixtime(bigint unixtime\, string format\)返回值: string说明: 转化UNIX时间戳(从19700101 00:00:00 UTC到指定时间的秒数)到当前时区的时间格式举例:hive   selec
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Stella981 Stella981
3年前
JavaScript常用函数
1\.字符串长度截取functioncutstr(str,len){vartemp,icount0,patrn/^\x00\xff/,strre"";for(vari
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
11个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这