Java深入学习系列之值传递Or引用传递?

简介:


我们来看一个新手甚至写了多年Java的朋友都可能不是十分确定的问题:

在Java方法传参时,究竟是引用传递还是值传递?

为了说明问题, 我给出一个非常简单的class定义:


 
 
  1. public class Foo { 
  2.   String attribute; 
  3.   Foo(String s) { 
  4.     this.attribute = s; 
  5.   } 
  6.   void setAttribute(String s) { 
  7.     this.attribute = s; 
  8.   } 
  9.   String getAttribute() { 
  10.     return this.attribute; 
  11.   } 
  12. }  

下面在阐明观点时,可能会多次用到该类。

关于Java里值传递还是引用传递,至少从表现形式上来看,两种观点都有支撑的论据。下面我来一一分析:

观点1:引用传递

理由如下:先看一段代码


 
 
  1. public class Main { 
  2.   public static void modifyReference(Foo c){ 
  3.     c.setAttribute("c"); // line DDD 
  4.   } 
  5.  
  6.   public static void main(String[] args) { 
  7.     Foo fooRef = new Foo("a"); // line AAA 
  8.     modifyReference(fooRef); // line BBB 
  9.     System.out.println(fooRef.getAttribute()); // 输出 c 
  10.   } 
  11. }  

上述示例,输出结果为"c",而不是"c"。

我们在line AAA处新创建了一个Object Foo并将其引用fooRef在line BBB处传给了方法modifyReference()的参数cRef, 该方法内部处理后,fooRef指向的Object中的值从"a"变成了"c", 而引用fooRef还是那个引用, 因此,我们是否可以认为,在line BBB处发生了引用传递?

先留着疑问,我们继续往下看。

观点2:值传递

继续看一段代码


 
 
  1. public class Main { 
  2.   public static void changeReference(Foo aRef){ 
  3.     Foo bRef = new Foo("b"); 
  4.     aRef = bRef;   // line EEE 
  5.   } 
  6.    
  7.   public static void main(String[] args) { 
  8.     Foo fooRef = new Foo("a"); // line AAA 
  9.     changeReference(fooRef); // line BBB 
  10.     System.out.println(fooRef.getAttribute()); // 输出 a 
  11.   } 
  12. }  

上述示例,输出结果为"a", 而不是"b"。

我们在line AAA处新创建了一个Object Foo并将其引用fooRef在line EEE处传给了方法changeReference()的参数aRef, 该方法内部引用aRef在line DDD处被重新赋值。如果是引用传递,那么引用aRef在line EEE处已经被指向了新的Object, 输出应该为"b"才对,事实上是怎样的呢?事实上输出了"b",也就是说changeReference()方法改变了传入引用所指对象的值。

观点1和观点2的输出结果多少会让人有些困惑,别急,我们继续往下看。

深入分析

为了详细分析这个问题,把上述两段代码合起来:


 
 
  1. public class Main { 
  2.   public static void modifyReference(Foo cRef){ 
  3.     cRef.setAttribute("c"); // line DDD 
  4.   } 
  5.   public static void changeReference(Foo aRef){ 
  6.     Foo bRef = new Foo("b"); // line FFF 
  7.     aRef = bRef;   // line EEE 
  8.   } 
  9.    
  10.   public static void main(String[] args) { 
  11.     Foo fooRef = new Foo("a"); // line AAA 
  12.     changeReference(fooRef); // line BBB 
  13.     System.out.println(fooRef.getAttribute()); // 输出 a 
  14.      
  15.     modifyReference(fooRef); // line CCC 
  16.     System.out.println(fooRef.getAttribute()); // 输出 c 
  17.      
  18.  
  19.   } 
  20. }  

下面来深入内部来详细分析一下引用和Object内部的变化。来看下面图示:

① Line AAA, 申明一个名叫fooRef,类型为Foo的引用,并见其分配给一个新的包含属性值为"f"的对象,该对象类型为Foo。


 
 
  1. Foo fooRef = new Foo("a"); // line AAA 

 

② Line DDD, 方法内部,申明了一个Foo类型的名为aRef的引用,且aRef被初始化为null。


 
 
  1. void changeReference(Foo a); 

③ Line CCC, changeReference()方法被调用后,引用aRef被分配给fooRef指向的对象。


 
 
  1. changeReference(fooRef); 

④ Line FFF, 申明一个名叫bRef,类型为Foo的引用,并见其分配给一个新的包含属性值为"b"的对象,该对象类型为Foo。


 
 
  1. Foo bRef = new Foo("b"); 

 

⑤ Line EEE, 将引用aRef重新分配给了包含属性"b"的对象。此处注意,并非将fooRef重新分配,而是aRef。


 
 
  1. aRef = bRef; 

 

⑥ Line CCC, 调用方法modifyReference(Foo cRef)后,新建了一个引用cRef并将之分配到包含该属性"f"的对象上,该对象同时被两个引用fooRef和cRef指向着。


 
 
  1. modifyReference(fooRef);  

⑦ Line DDD, cRef.setAttribute("c");将会改变cRef引用指向的包含属性"f"的对象,而该对象同时被引用fooRef指向着。


 
 
  1. cRef.setAttribute("c"); 

 

此时引用fooRef指向的对象内部属性值"f"也被重新设置为"c"。

总结

Java内部方法传参不是引用传递,而是引用本身的"值"的传递,归根结底还是值传递。将一个对象的引用fooRef传给方法的形参newRef,将给该对象新增了一个引用,相当于多了一个alias。我们可以通过这个原引用fooRef,或这是方法参数里的新引用newRef去访问、操作原对象,也可以改变参数里的引用newRef本身的值,却无法改变原引用fooRef的值。


作者:夏磊洲
来源:51CTO

相关文章
|
3月前
|
编解码 Oracle Java
java9到java17的新特性学习--github新项目
本文宣布了一个名为"JavaLearnNote"的新GitHub项目,该项目旨在帮助Java开发者深入理解和掌握从Java 9到Java 17的每个版本的关键新特性,并通过实战演示、社区支持和持续更新来促进学习。
99 3
|
3月前
|
XML Java 编译器
Java学习十六—掌握注解:让编程更简单
Java 注解(Annotation)是一种特殊的语法结构,可以在代码中嵌入元数据。它们不直接影响代码的运行,但可以通过工具和框架提供额外的信息,帮助在编译、部署或运行时进行处理。
105 43
Java学习十六—掌握注解:让编程更简单
|
2月前
|
Java 大数据 API
14天Java基础学习——第1天:Java入门和环境搭建
本文介绍了Java的基础知识,包括Java的简介、历史和应用领域。详细讲解了如何安装JDK并配置环境变量,以及如何使用IntelliJ IDEA创建和运行Java项目。通过示例代码“HelloWorld.java”,展示了从编写到运行的全过程。适合初学者快速入门Java编程。
|
3月前
|
存储 SQL 小程序
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
这篇文章详细介绍了Java虚拟机(JVM)的运行时数据区域和JVM指令集,包括程序计数器、虚拟机栈、本地方法栈、直接内存、方法区和堆,以及栈帧的组成部分和执行流程。
45 2
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
|
2月前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。
|
3月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
61 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
3月前
|
前端开发 Java 应用服务中间件
Javaweb学习
【10月更文挑战第1天】Javaweb学习
39 2
|
3月前
|
存储 安全 Java
【用Java学习数据结构系列】探索顺序表和链表的无尽秘密(附带练习唔)pro
【用Java学习数据结构系列】探索顺序表和链表的无尽秘密(附带练习唔)pro
30 3
|
3月前
|
存储 安全 Java
【用Java学习数据结构系列】探索栈和队列的无尽秘密
【用Java学习数据结构系列】探索栈和队列的无尽秘密
37 2
|
3月前
|
存储 Java 编译器
【用Java学习数据结构系列】初识泛型
【用Java学习数据结构系列】初识泛型
25 2