秒懂Java为什么只有值传递
在Java语言中,数据类型分为基本数据类型和引用数据类型。
基本数据类型(如int、double、char等)的值直接保存在栈上。这些类型的变量在栈内存中有固定的大小,并且值是直接存储在这些变量中的,数据的传递为值传递,这个好理解。以下以引用数据类型来讲解。
引用和实例化对象
比如new一个对象的代码:等号左边的person为引用;等号的右边new Person()为实例化对象。
Person person = new Person();
引用 = 实例化对象
引用和实例化对象在内存中,分别保存在栈和堆中。引用会保存着实例化对象的地址,从而可以通过引用来获取到具体的实例化对象保存在哪里。
关系如图:

值传递和引用传递
顾名思义,值传递是把“值”传递到方法中,而引用传递是把“引用”传递到方法中。
在Java 的参数传递方式中,只有值传递。对于引用对象也是值传递,而这个值是引用的值(即堆地址或句柄)被拷贝传递到方法中。
文字太抽象了,看图和代码:


对比两者
值传递
源引用地址:0x0a --> 对象地址:0x10
传递时:新引用地址:0x0b --> 对象地址:0x10
引用传递
源引用地址:0x0a --> 对象地址:0x10
传递时:方法中仍是引用地址:0x0a --> 对象地址:0x10
值传递:是复制一份“引用”传给方法的形参,person 和 person2 是两个不同的栈内存地址(如 0x0a和0x0b)
引用传递:是将实参的引用直接传给方法的形参,person 和 person2 实际共享了相同的栈内存地址(如 0x0a)
两者有着本质的区别!对比两者的行为和带来的影响。
对比示意图
| 行为 | 值传递(Java 的实际行为) | 引用传递(假设机制) |
|---|---|---|
| 方法参数接收到的值 | 引用地址的副本(如 0x0b) |
原始引用地址本身(如 0x0a) |
| 引用的改变影响范围 | 改变方法内的引用指向,不影响原始引用 | 改变引用指向会影响原始引用 |
| 对象属性的修改 | 通过引用修改对象属性,会影响原始对象 | 通过引用修改对象属性,会影响原始对象 |
代码验证
测试代码:测试引用的改变影响范围和对象属性的修改
public class ValuePassDemo {
public static void main(String[] args) {
Person person = new Person();
person.setAge(18);
person.setName("Denny");
// 初始值
System.out.println("地址:"+ Integer.toHexString(person.hashCode()) + ">>>" + person);
modifyReference(person);
// 是否会被上一个方法修改值
System.out.println("地址:"+ Integer.toHexString(person.hashCode()) + ">>>" + person);
modifyReference2(person);
// 是否会被上一个方法修改值
System.out.println("地址:"+ Integer.toHexString(person.hashCode()) + ">>>" + person);
}
/**
* 形参和实参引用指向的实例化对象是同一个
* 实例化对象的值被任意一边修改时,都会改变
*/
public static void modifyReference(Person person2) {
person2.setAge(28);
person2.setName("Jack");
System.out.println("地址:"+ Integer.toHexString(person2.hashCode()) + ">>>" + person2);
}
/**
* 如果是引用传递,
* 那实参引用 person 等于形参引用 person2,
* 那么引用 person2 的指向被改变的话,形参引用 person也会指向新的实例化对象
* 如果不成立,那就是值传递,引用person2 只是引用person的拷贝,而非本身给了它
*/
public static void modifyReference2(Person person2) {
person2 = new Person();
person2.setAge(20);
person2.setName("apple");
System.out.println("地址:"+ Integer.toHexString(person2.hashCode()) + ">>>" + person2);
}
}
测试结果:

方法内部修改引用的指向
调用modifyReference2方法,会给形参变量赋一个新的实例化对象的情况,
如果是值传递,形参和实参分别指向不同的实例化对象,如图:

如果是引用传递,形参和实参都指向相同的实例化对象,而原来的实例化对象就没有引用指向。
如图:

原来的实例化对象没有引用指向,会导致内存泄漏,用C++的语言描述:按引用传递时,并且在方法内修改引用指向新new的对象时,需要手动释放内存。
void myFunction(Person* obj)是按值传递。void myFunction(Person*& obj)是引用传递。
void myFunctionWithReference(Person*& obj) {
delete obj; // 先释放原对象的内存
obj = new Person(20); // 重新分配新的对象,并让 obj 指向它
}
为什么Java只有值传递
个人觉得Java 选择只有值传递的参数传递机制(pass-by-value)目的应该包含:内存安全性、简化内存管理、保持语言行为一致性和语言简单易用。
这也是Java语言的优点,弱化对内存操作的概念,让这门语言更加简洁易用;同时这也是Java语言的缺点,降低了灵活性,无法直接通过方法修改引用变量的指向。

秒懂Java为什么只有值传递的更多相关文章
- Java 中的值传递和参数传递
Java中没有指针,所以也没有引用传递了,仅仅有值传递不过可以通过对象的方式来实现引用传递 类似java没有多继承 但可以用多次implements 接口实现多继承的功能 值传递:方法调用时,实际参数 ...
- Java中的值传递和引用传递
这几天一直再纠结这个问题,今天看了这篇文章有点思路了,这跟C++里函数参数为引用.指针还是有很大区别. 当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里 ...
- java是通过值传递,也就是通过拷贝传递——通过方法操作不同类型的变量加深理解(勿删)
head first java里写到“java是通过值传递的,也就是通过拷贝传递”,由此得出结论,方法无法改变调用方传入的参数.该怎么理解呢? 看例子: public class Test1 { pu ...
- Java中的值传递
1.先比较下字符串的比较 == 代表全等于 值和地址(存放地址) 全部相等于. equals 值等于== 和 equals的区别 列如下面的 如果name1==name2是等于的 然而name1==n ...
- 为什么说Java中只有值传递
本文转载自公众号 Hollis 对于初学者来说,要想把这个问题回答正确,是比较难的.在第二天整理答案的时候,我发现我竟然无法通过简单的语言把这个事情描述的很容易理解,遗憾的是,我也没有在网上找到哪篇文 ...
- 一道笔试题来理顺Java中的值传递和引用传递
题目如下: private static void change(StringBuffer str11, StringBuffer str12) { str12 = str11; str11 = ...
- java中的值传递和引用传递有什么区别呀?
值传递: (形式参数类型是基本数据类型和String):方法调用时,实际参数把它的值传递给对应的形式参数,形式参数只是用实际参数的值初始化自己的存储单元内容,是两个不同的存储单元,所以方法执行中形式参 ...
- 为什么说Java中只有值传递(转载)
出处:https://www.hollischuang.com/archives/2275 关于这个问题,在StackOverflow上也引发过广泛的讨论,看来很多程序员对于这个问题的理解都不尽相同, ...
- 为什么说Java中只有值传递?
一.为什么说Java中只有值传递? 对于java中的参数传递方式中是否有引用传递这个话题,很多的人都认为Java中有引用传递,但是我个人的看法是,Java中只有值传递,没有引用传递. 那么关于对象的传 ...
- JAVA中只有值传递
今天,我在一本面试书上看到了关于java的一个参数传递的问题: 写道 java中对象作为参数传递给一个方法,到底是值传递,还是引用传递? 我毫无疑问的回答:“引用传递!”,并且还觉得自己对java的这 ...
随机推荐
- Serilog文档翻译系列(四) - 结构化数据
Serilog 是一种序列化器.在许多情况下,它具有良好的默认行为,能够满足其目的,但有时也需要指示 Serilog 如何存储附加到日志事件上的属性. Serilog 使用一些不寻常的术语来指代 .N ...
- FCA-FineBI最新版考试答案,全全全!!!
FCA-FineBI最新版考试答案,全全全!!!-CSDN博客同博客 Part.1:判断 第1题 判断题 「TODATE」函数或者「DATE」函数,可以将文本字段或数值字段转变成时间类型的字段.(得 ...
- JavaScript习题之填空题
1. JavaScript有两种引⽤数据类型:__数组___.__对象__.2. Javascript通过__setTimeout___延迟指定时间后,去执⾏某程序.3. Javascript⾥Str ...
- MyBatis——案例——查询-查询详情
查询-查询详情 (根据id获取商品全部信息(即商品对象)) 1.编写Mapper接口方法:Brand selectById(int id); 2.编写SQL ...
- java使用正则表达式验证手机号和电话号码和邮箱号码的方法
验证手机号 我国的手机号一般是以1开头,后面跟着10位数字.因此,可以用如下正则表达式: public static boolean isValidPhoneNumber(String phoneNu ...
- constexpr声明 常量表达式
constexpr 是 C++ 中用于声明 常量表达式 的关键字,表示一个变量或函数的值在编译时就可以确定,而不是在运行时计算. 这对优化非常有用,因为编译器可以直接将结果嵌入到程序中,减少运行时 ...
- glance对接ceph
目录 glance对接ceph 1. 上传镜像 2. 对接ceph 2.1 创建池 2.2 创建用户 2.3 下发ceph文件 2.4 修改globals文件 2.5 更新glance配置 3. 上传 ...
- 七张图看懂 Linux profiling 机制
1 图 1 Linux profiling 手段一览 软件埋点: 手动埋点:主动调用 trace 函数来实现埋点. Android systrace 即是这样一个例子,如图 2 和 图 3 所示 自动 ...
- 常见的mysql 函数 字符串函数
1. concat (s1,s2,....sn) 字符串拼接,将 s1,s2,... sn 拼接成一个字符串 : 2. lower(str) 将字符串全部转换成小写 3. upper(str) 将字符 ...
- 云原生的 WebAssembly 能取代 Docker 吗?
WebAssembly 是一个可移植.体积小.加载快并且兼容 Web 的全新格式.由于 WebAssembly 具有很高的安全性,可移植性,效率和轻量级功能,因此它是应用程序安全沙箱方案的理想选择.现 ...