java的值传递

先看以下代码:
class Person {
private int age;
private String name;
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public static void main(String[] args) {

Person person = new Person();//---1
Person p2 = new Person();
person.setAge(12);
person.setName("lj");
test.changePerson(person,p2);//---2 该处person不同于1处的person,而是前者的一个拷贝,他们指向堆中同一个对象
System.out.println("in main...");
System.out.println("person.age:" + person.getAge());
System.out.println("person.name:" + person.getName());
}
void changePerson(Person p1,Person p2){
p2.setAge(13);
p2.setName("yb");
p1 = p2;
System.out.println("in changePerson...");
System.out.println("person.age:" + p1.getAge());
System.out.println("person.name:" + p1.getName());
}
}

运行结果:
in changePerson...
person.age:13
person.name:yb
in main...
person.age:12
person.name:lj
也许会很奇怪,java对象在方法里传递时不是引用传递么?为什么传过去的person在changePerson()方法体内把引用指向了p2,但回到主方法后又把引用指向了原来的对象
原因如下:

主函数中: person--> [age:12 name:lj] <--person’

person'即为test.changePerson(person,p2); 中的person,它与前面的person具有相同的引用地址,因此java中的“对象变量”的传递叫做“引用传递”。但就其实质而言,还是值传递,只不过这个值是引用(或者说地址)而已。

changePerson函数中:person对象的拷贝p1又把引用指向了另一个对象p2,所以会打印出新对象的信息,而当这个函数结束,p1的生命周期也就结束了。
再次回到主函数时,person对象指向原来的对象的事实没有改变,所以还是打印出原来对象的信息

如果还有一个函数changePerson2如下:
void changePerson2(Person p){
p.setAge(100);
p.setName("ljyb");
}

则在主函数增加如下代码:
test.changePerson2(person);
System.out.println("in changePerson...");
System.out.println("person.age:" + person.getAge());
System.out.println("person.name:" + person.getName());
结果为:
after invoking changePerson2...
person.age:100
person.name:ljyb

此时的传递方式与上面相同,changePerson2中的p变量也是主函数中person变量的一份拷贝,他们指向同一个对象。p改变了这个对象的值,也就改变了person指向对象的值。
当changePerson2函数结束时,p的生命周期也结束了,但它在changePerson2内做的操作却被保存下来了。
此时打印person的信息就变成修改后的信息了。

我还找到一些相关资料,附下
下面的文字来自一个哥们的文章中的评论,他写的这篇文章也是关于java值传递的
网址:http://zangweiren.iteye.com/blog/214369

有些争论没有意义,其实你知道传值和传引用由什么区别就可以了,但不要自以为是的认为就是别人混淆了。
“有一种说法是当一个对象或引用类型变量被当作参数传递时,也是值传递,这个值就是对象的引用,因此JAVA中只有值传递,没有引用传递。这种说法显然是混淆了值和引用的概念。”
我看到的很多书中都是只有pass by value的概念。
James Gosling,我想他对Java语言里的一些概念还是有话语权的,《The Java Programming Language》2.6.5. Parameter Values一节中,他的说法是 :
[quote]All parameters to methods are passed "by value." In other words, values of parameter variables in a method are copies of the values the invoker specified as arguments.。。。。。You should note that when the parameter is an object reference, it is the object reference not the object itself that is passed "by value." Thus, you can change which object a parameter refers to inside the method without affecting the reference that was passed. But if you change any fields of the object or invoke methods that change the object's state, the object is changed for every part of the program that holds a reference to it. [/quote]这里说的很清楚了,方法的所有参数都是值传递。

别的书里也有提及 ,Core Java 卷一中也有类似的说法,举了一些例子。
Thinking In Java中也提到过,具体什么地方记不清楚了。
从编译原理的概念来说,有传值,传地址,传名,传结果等。这里的传地址明显和java的传引用不一样。C++中引入了传引用,但是C++的传引用有个明显的特点,引用一旦创建不能修改,所以引用不能指向新的对象。所以不会出现java这种令人混淆的概念。所以还是多用Java中的术语,区分pass by value中不同的情况,而不是自己引入pass by reference 的概念让人们更加混淆。
豌豆代理(又称豌豆 IP)是一款一站式国内代理 IP 服务平台,主打高匿名、低延迟、高可用的 IP 资源,支持 HTTP/HTTPS/SOCKS5 协议,适配 Windows、Mac、Android、iOS 多平台。 多类型 IP 资源与高覆盖节点 提供动态住宅 IP、静态独享 IP、数据中心 IP,覆盖全国 200 + 城市,可用率 99%+;支持省市精准选择或全国混拨,适配不同业务合规与稳定性需求。 使用:在客户端 “节点 / 线路” 页,按城市 / 类型筛选,一键连接目标 IP,适合爬虫、电商多账号运营等场景。 秒级 IP 切换与灵活调度 支持手动一键切换、秒级动态切换(切换速度低至 100ms)、定时切换(自定义时长),并自动过滤重复 IP,避免重复使用导致风险。 使用:在 “设置” 中开启 “自动切换” 并设时间间隔,或按 Ctrl+Q 快捷键一键换 IP,适配反爬虫、批量测试等高频切换场景。 全协议支持与多端适配 兼容 HTTP/HTTPS/SOCKS5 主流代理协议,可对接浏览器、爬虫脚本、客户端软件;支持 Windows、Mac、安卓、iOS 多端同步使用,跨设备无缝切换。 使用:在客户端 “协议设置” 选择对应协议,生成代理地址与端口,直接填入目标软件即可生效。 隐私安全与数据加密 自研传输加密技术保护数据传输,搭配高匿名 IP 隐藏真实地址,同时支持自动清除 Cookie / 缓存,降低隐私泄露与追踪风险。 使用:在 “安全设置” 中开启 “数据加密” 与 “自动清理缓存”,公共 WiFi 环境下优先启用,提升隐私防护等级。 智能筛选与稳定网络优化 系统自动筛选低延迟、高可用 IP,过滤失效 / 重复地址;依托自建纯净机房与独享带宽,搭配 BGP 多线接入,保障连接稳定性与速度。 使用:无需手动配置,客户端默认智能匹配合适节点,复杂网络环境可在 “网络
在网络高速发展的时代,众多的软件被开发出来,给用户带来了很大的选择余地,而且人们越来越追求更个性的需求。在这种时代背景下,商家只能以用户为导向,以商品的持续创新作为商家最重要的事项。 在新发展的时代,人们对幼儿资源互助共享平台越来越重视,才能实现幼儿资源互助共享平台的有效发挥,本文将通过幼儿资源互助共享平台的信息,分析在日常生活中对幼儿资源互助共享平台存在哪些问题探讨出进一步提升效率,管理能力的对策。 系统采用了Java技术,将所有模块采用以浏览器交互的模式,选择MySQL作为系统的数据库,来进行系统的设计。基本实现了幼儿资源互助共享平台应有的主要功能模块,本系统有管理员:首页、个人中心、用户管理、卖家管理、咨询师管理、萌宝信息管理、幼儿知识管理、保姆推荐管理、音频资源管理、二手商品管理、商品分类管理、资源分类管理、交流论坛、系统管理,用户;首页、个人中心、萌宝信息管理、保姆推荐管理、音频资源管理,卖家;首页、个人中心、二手商品管理、订单管理,咨询师;首页、个人中心、幼儿知识管理,前台首页;首页、萌宝信息、幼儿知识、保姆推荐、音频资源、二手商品、交流论坛、个人中心、后台管理、购物车等功能。 对系统进行测试后,改善了程序逻辑和代码。同时确保系统中所有的程序都能正常运行,所有的功能都能操作,本系统的开发获取幼儿资源互助共享平台信息能够更加方便快捷,同时也使幼儿资源互助共享平台信息变的更加系统化、有序化。系统界面较友好,易于操作。
<think>首先,用户的问题是“java值传递问题”。这是在问Java中的值传递机制。Java值传递语言,这意味着在方法调用时,参数是原始值的副本。对于基本类型,传递的是值的副本;对于对象引用,传递的是引用的副本。 我需要详细解释Java值传递机制,并提供代码示例来演示。代码应该用Java编写,并使用Markdown格式包裹。 回答结构: 1. **回答问题**:解释Java值传递概念。 2. **给出解释**:详细说明为什么Java值传递,并提供代码示例和解释。 3. **相关代码**:提供Java代码来展示值传递的行为。 4. **提出相关问题**:在最后,按照指定格式提出相关问题。 ### 关键点: - **基本类型(如int, double)**:传递的是值的副本。方法内部修改不会影响原始值。 - **对象引用**:传递的是引用的副本。这意味着方法内部可以修改对象的状态,但不能改变原始引用(即不能重新分配对象)。 代码示例: - 示例1:基本类型值传递。 - 示例2:对象引用值传递。 ### 回答大纲: 1. **介绍**:Java值传递语言。所有参数都是按值传递。 2. **基本类型示例**:展示一个方法修改基本类型参数,但原始值不变。 3. **对象引用示例**:展示一个方法修改对象的状态(如字段),但尝试重新分配引用时,原始引用不变。 4. **解释**:为什么Java值传递,以及常见误解。 5. **相关问题**:按照格式提出4个相关问题。 ### 代码示例: **示例1:基本类型值传递** ```java public class ValuePassingDemo { public static void main(String[] args) { int number = 10; System.out.println("Before method call: " + number); // Output: 10 modifyPrimitive(number); System.out.println("After method call: " + number); // Output: 10 } public static void modifyPrimitive(int num) { num = 20; System.out.println("Inside method: " + num); // Output: 20 } } ``` **示例2:对象引用值传递** ```java class MyObject { int value; MyObject(int value) { this.value = value; } } public class ReferencePassingDemo { public static void main(String[] args) { MyObject obj = new MyObject(10); System.out.println("Before method call: " + obj.value); // Output: 10 modifyReference(obj); System.out.println("After method call: " + obj.value); // Output: 20 // But if we try to reassign the reference reassignReference(obj); System.out.println("After reassign: " + obj.value); // Output: 20 (unchanged) } public static void modifyReference(MyObject param) { param.value = 20; // Modifies the object's state } public static void reassignReference(MyObject param) { param = new MyObject(30); // Reassigns the copy of the reference, not the original System.out.println("Inside reassign: " + param.value); // Output: 30 } } ``` ### 解释: - 在第一个示例中,`modifyPrimitive` 方法接收 `number` 的副本,修改副本不影响原始值。 - 在第二个示例中,`modifyReference` 方法接收引用的副本,但通过该引用修改了对象的状态,所以原始对象被改变。然而,`reassignReference` 方法尝试重新分配引用,但由于是副本,原始引用不变。 ### 相关问题: 按照格式,提出4个相关问题: 1. Java中基本类型和引用类型在传递时的区别。 2. 如何模拟引用传递或指针传递在Java中。 3. Java中的String对象为什么在方法中修改不会影响原始值。 4. 值传递和引用传递在其他语言(如C++)中的区别。 最后,确保回答以指定格式结束。</think>### Java 值传递问题详解 在 Java 中,所有参数传递都是**值传递(Pass by Value)**。这意味着: - **基本类型(如 int, double, boolean)**:传递的是值的副本。方法内部修改不会影响原始变量。 - **引用类型(如对象、数组)**:传递的是引用的副本(即内存地址的副本)。方法内部可以修改对象的状态,但**不能改变原始引用的指向**(例如,不能通过参数重新分配对象)。 以下代码示例演示了 Java值传递行为: #### 示例 1: 基本类型值传递 ```java public class PrimitivePassingDemo { public static void main(String[] args) { int num = 10; System.out.println("Before modify: " + num); // 输出: 10 modifyPrimitive(num); System.out.println("After modify: " + num); // 输出: 10 (未改变) } // 方法接收基本类型参数的副本 public static void modifyPrimitive(int value) { value = 20; // 修改副本,不影响原始变量 System.out.println("Inside method: " + value); // 输出: 20 } } ``` **解释**: - `num` 是基本类型,传递时创建副本 `value`。 - 方法内修改 `value` 不影响原始 `num`,因为两者是独立的内存空间。 #### 示例 2: 引用类型值传递 ```java class MyObject { int data; MyObject(int data) { this.data = data; } } public class ReferencePassingDemo { public static void main(String[] args) { MyObject obj = new MyObject(10); System.out.println("Before modify: " + obj.data); // 输出: 10 modifyObjectState(obj); // 修改对象状态 System.out.println("After state modify: " + obj.data); // 输出: 20 reassignObject(obj); // 尝试重新分配引用 System.out.println("After reassign: " + obj.data); // 输出: 20 (未改变) } // 1. 修改对象状态:通过引用副本操作原始对象 public static void modifyObjectState(MyObject param) { param.data = 20; // 修改对象字段,影响原始对象 } // 2. 重新分配引用:只影响副本,不影响原始引用 public static void reassignObject(MyObject param) { param = new MyObject(30); // 副本指向新对象,原始引用不变 System.out.println("Inside reassign: " + param.data); // 输出: 30 } } ``` **解释**: - **`modifyObjectState` 方法**: `param` 是原始引用 `obj` 的副本,两者指向同一对象。修改 `param.data` 会改变原始对象的状态。 - **`reassignObject` 方法**: `param = new MyObject(30)` 只让副本指向新对象,原始引用 `obj` 仍指向原对象。因此原始对象未被替换。 #### 关键结论: 1. **值传递本质**:Java 始终传递值的副本(基本类型是数据副本,引用类型是地址副本)。 2. **常见误解**: - 修改引用类型对象的状态 ≠ 改变原始引用。 - 无法通过方法参数让原始引用指向新对象(需通过返回值实现)。 3. **与引用传递的区别**: 真正的引用传递(如 C++ 的 `&`)允许方法直接修改原始变量。Java 不支持此机制。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值