Java 值传递与引用传递的一些误区

本贴最后更新于 2685 天前,其中的信息可能已经时移世易

关于 Java 值传递与引用传递的讨论有很多,先来了解一下值传递和引用传递。

值传递:是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。

引用传递:是指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。

概念知道了,接下来看问题:

public static void main(String[] args) {
	int a = 0;
	char[] c = {'a', 'b', 'c'}; 
	change(a, c);
	System.out.println("a:" + a);
	System.out.println("c[0]:" + c[0]);
}

private static void change(int a, char[] c) {
	a = 2;
	c[0] = '0';
}

输出结果为:

a:0
c[0]:0

这种写法很容易让人产生误解,认为 change 方法里直接对 a 重新赋值,直接对 c[0]重新赋值,我们换个写法:

public static void main(String[] args) {
	int a = 0;
	char[] c = {'a', 'b', 'c'}; 
	change(a, c);
	System.out.println("a:" + a);
	System.out.println("c[0]:" + c[0]);
}

/**
 * 为避免混淆,换个参数名
 */
private static void change(int aa, char[] cc) {
	aa = 2;
	cc[0] = '0';
}

输出结果为:

a:0
c[0]:0

change 方法里对 aa 的修改并没有影响到 main 方法里 a 的值,所以第一个参数是值传递没有问题。但是对 cc[0]的修改却影响到了 main 方法里 c[0]的值,这是否说明第二个参数是引用传递呢?我们再来修改一下代码:

public static void main(String[] args) {
	int a = 0;
	char[] c = {'a', 'b', 'c'}; 
	change(a, c);
	System.out.println("a:" + a);
	System.out.println("c[0]:" + c[0]);
}

/**
 * 为避免混淆,换个参数名
 */
private static void change(int aa, char[] cc) {
	aa = 2;
	cc = null;
}

输出结果为:

a:0
c[0]:a

由此可知,对 cc 的修改其实并不会影响 main 方法里的 c。那么刚才为什么可以影响呢?因为调用 change 方法时,我们把 c 当做参数传了进去,而在 change 方法里,又有一个形参 cc,此时这两个引用同时指向同一个内存地址,也就是说,这时候不管用哪个引用去操作这块内存区域的数组,都会导致数组发生变化,但是请注意,这里说的是对数组进行操作,如果在 change 方法里把 cc 赋值为 null,这个就不是对数组进行操作了,而是把 cc 由指向数组变为指向 null,相当于有两个变量,一个是 c,一个是 cc,不管把 cc 赋值成什么,都与 c 无关。所以,这里并没有引用传递,也是值传递。

再来看一个进阶版:

public static void main(String[] args) {
	String str = "abc";
	char[] c = {'a', 'b', 'c'}; 
	change(str, c);
	System.out.println("str:" + str);
	System.out.println("c[0]:" + c[0]);
}

/**
 * 为避免混淆,换个参数名
 */
private static void change(String s, char[] cc) {
	s = "xyz";
	cc[0] = '0';
}

输出结果为:

str:abc
c[0]:0

两个参数都是引用数据类型,怎么会一个改变了,一个没有改变呢?有些解释是,string 是 final 修饰的,不能改变,所以虽然 string 是引用数据类型,但是它是个特例,是值传递,但是数组能改变,是引用传递。通过上面的分析,我们已经知道,不管是引用数据类型,还是基本数据类型,其参数传递都是值传递,但是引用数据类型是可以改变其属性的,那为什么 string 没有改变,难道真的是特例吗?

我们都知道,字符串其实也就相当于是复杂点的 char 数组,仔细观察会发现,change 方法里面对字符串 s 的操作其实是相当于改变其引用,并没有对字符串里的数组进行属性改变,而对数组 cc 的操作却是改变数组的属性,也就是说,这两个操作根本不在一个层面。要么就都改变引用,要么就都改变属性,操作保证一致,才能够说明问题。

修改代码,使 change 改变属性:

public static void main(String[] args) throws Exception {
	String str = "abc";
	char[] c = {'a', 'b', 'c'}; 
	change(str, c);
	System.out.println("str:" + str);
	System.out.println("c[0]:" + c[0]);
}

/**
 * 为避免混淆,换个参数名
 */
private static void change(String s, char[] cc) throws Exception {
	Field valueFieldOfString = String.class.getDeclaredField("value");  
    valueFieldOfString.setAccessible(true);  
    char[] value = (char[]) valueFieldOfString.get(s);  
    value[0] = '0';
	cc[0] = '0';
}

输出结果为:

str:0bc
c[0]:0

通过反射获取到 string 的私有 char 数组属性,并修改数组的第一个元素,说明 string 并不是不能修改,也不是什么特例。change 方法影响到了 main 方法里的原始变量,但是原因是我们上面分析的,只是因为两个引用都指向同一个内存地址而已,本质是值传递,并不是引用传递。

修改代码,使 change 改变引用:

public static void main(String[] args) {
	String str = "abc";
	char[] c = {'a', 'b', 'c'}; 
	change(str, c);
	System.out.println("str:" + str);
	System.out.println("c[0]:" + c[0]);
}

/**
 * 为避免混淆,换个参数名
 */
private static void change(String s, char[] cc) {
	s = null;
	cc = null;
}

输出结果为:

str:abc
c[0]:a

并没有影响 main 方法中原始变量,值传递。

通过上面的一系列分析,我们可以确定,Java 中不管是基本数据类型,还是引用数据类型,都只存在值传递,不存在引用传递,更不存在什么特例。

  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3186 引用 • 8212 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
  • someone

    写的还可以,不错[em01][em01]

  • someone

    [em00]

  • someone

    互相学习~

  • someone

    [em01]

  • someone

    [em00]

  • someone

    测试下评论url。

  • someone

    评论url有问题,折腾了好久。。

请输入回帖内容 ...

推荐标签 标签

  • 阿里巴巴

    阿里巴巴网络技术有限公司(简称:阿里巴巴集团)是以曾担任英语教师的马云为首的 18 人,于 1999 年在中国杭州创立,他们相信互联网能够创造公平的竞争环境,让小企业通过创新与科技扩展业务,并在参与国内或全球市场竞争时处于更有利的位置。

    43 引用 • 221 回帖 • 129 关注
  • GitLab

    GitLab 是利用 Ruby 一个开源的版本管理系统,实现一个自托管的 Git 项目仓库,可通过 Web 界面操作公开或私有项目。

    46 引用 • 72 回帖
  • 锤子科技

    锤子科技(Smartisan)成立于 2012 年 5 月,是一家制造移动互联网终端设备的公司,公司的使命是用完美主义的工匠精神,打造用户体验一流的数码消费类产品(智能手机为主),改善人们的生活质量。

    4 引用 • 31 回帖
  • iOS

    iOS 是由苹果公司开发的移动操作系统,最早于 2007 年 1 月 9 日的 Macworld 大会上公布这个系统,最初是设计给 iPhone 使用的,后来陆续套用到 iPod touch、iPad 以及 Apple TV 等产品上。iOS 与苹果的 Mac OS X 操作系统一样,属于类 Unix 的商业操作系统。

    84 引用 • 139 回帖 • 1 关注
  • WebClipper

    Web Clipper 是一款浏览器剪藏扩展,它可以帮助你把网页内容剪藏到本地。

    3 引用 • 9 回帖 • 2 关注
  • Netty

    Netty 是一个基于 NIO 的客户端-服务器编程框架,使用 Netty 可以让你快速、简单地开发出一个可维护、高性能的网络应用,例如实现了某种协议的客户、服务端应用。

    49 引用 • 33 回帖 • 18 关注
  • abitmean

    有点意思就行了

    30 关注
  • 面试

    面试造航母,上班拧螺丝。多面试,少加班。

    325 引用 • 1395 回帖
  • Sphinx

    Sphinx 是一个基于 SQL 的全文检索引擎,可以结合 MySQL、PostgreSQL 做全文搜索,它可以提供比数据库本身更专业的搜索功能,使得应用程序更容易实现专业化的全文检索。

    1 引用 • 209 关注
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖
  • wolai

    我来 wolai:不仅仅是未来的云端笔记!

    2 引用 • 14 回帖
  • Docker

    Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的操作系统上。容器完全使用沙箱机制,几乎没有性能开销,可以很容易地在机器和数据中心中运行。

    490 引用 • 916 回帖 • 2 关注
  • 架构

    我们平时所说的“架构”主要是指软件架构,这是有关软件整体结构与组件的抽象描述,用于指导软件系统各个方面的设计。另外还有“业务架构”、“网络架构”、“硬件架构”等细分领域。

    142 引用 • 442 回帖
  • 倾城之链
    23 引用 • 66 回帖 • 139 关注
  • 笔记

    好记性不如烂笔头。

    308 引用 • 793 回帖
  • 设计模式

    设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

    200 引用 • 120 回帖 • 1 关注
  • HHKB

    HHKB 是富士通的 Happy Hacking 系列电容键盘。电容键盘即无接点静电电容式键盘(Capacitive Keyboard)。

    5 引用 • 74 回帖 • 465 关注
  • SQLite

    SQLite 是一个进程内的库,实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。SQLite 是全世界使用最为广泛的数据库引擎。

    5 引用 • 7 回帖
  • Spark

    Spark 是 UC Berkeley AMP lab 所开源的类 Hadoop MapReduce 的通用并行框架。Spark 拥有 Hadoop MapReduce 所具有的优点;但不同于 MapReduce 的是 Job 中间输出结果可以保存在内存中,从而不再需要读写 HDFS,因此 Spark 能更好地适用于数据挖掘与机器学习等需要迭代的 MapReduce 的算法。

    74 引用 • 46 回帖 • 561 关注
  • SpaceVim

    SpaceVim 是一个社区驱动的模块化 vim/neovim 配置集合,以模块的方式组织管理插件以
    及相关配置,为不同的语言开发量身定制了相关的开发模块,该模块提供代码自动补全,
    语法检查、格式化、调试、REPL 等特性。用户仅需载入相关语言的模块即可得到一个开箱
    即用的 Vim-IDE。

    3 引用 • 31 回帖 • 101 关注
  • Postman

    Postman 是一款简单好用的 HTTP API 调试工具。

    4 引用 • 3 回帖 • 3 关注
  • Shell

    Shell 脚本与 Windows/Dos 下的批处理相似,也就是用各类命令预先放入到一个文件中,方便一次性执行的一个程序文件,主要是方便管理员进行设置或者管理用的。但是它比 Windows 下的批处理更强大,比用其他编程程序编辑的程序效率更高,因为它使用了 Linux/Unix 下的命令。

    122 引用 • 73 回帖
  • 国际化

    i18n(其来源是英文单词 internationalization 的首末字符 i 和 n,18 为中间的字符数)是“国际化”的简称。对程序来说,国际化是指在不修改代码的情况下,能根据不同语言及地区显示相应的界面。

    8 引用 • 26 回帖
  • WebSocket

    WebSocket 是 HTML5 中定义的一种新协议,它实现了浏览器与服务器之间的全双工通信(full-duplex)。

    48 引用 • 206 回帖 • 347 关注
  • 星云链

    星云链是一个开源公链,业内简单的将其称为区块链上的谷歌。其实它不仅仅是区块链搜索引擎,一个公链的所有功能,它基本都有,比如你可以用它来开发部署你的去中心化的 APP,你可以在上面编写智能合约,发送交易等等。3 分钟快速接入星云链 (NAS) 测试网

    3 引用 • 16 回帖 • 2 关注
  • MySQL

    MySQL 是一个关系型数据库管理系统,由瑞典 MySQL AB 公司开发,目前属于 Oracle 公司。MySQL 是最流行的关系型数据库管理系统之一。

    677 引用 • 535 回帖
  • PHP

    PHP(Hypertext Preprocessor)是一种开源脚本语言。语法吸收了 C 语言、 Java 和 Perl 的特点,主要适用于 Web 开发领域,据说是世界上最好的编程语言。

    179 引用 • 407 回帖 • 489 关注