Java中所有参数都是值传递,基本类型传递值的副本,引用类型传递地址的副本;修改对象属性生效因操作同一堆对象,重新赋值不生效因仅改变形参指向。

Java中所有参数都是值传递,包括对象引用
Java没有引用传递,这是很多人误以为“对象是引用传递”的根源。所谓“值传递”,是指方法接收的是实参的副本——对基本类型是值的拷贝,对引用类型是引用地址的拷贝。这意味着:obj 和 obj2 如果指向同一对象,它们的地址值相同,但这两个变量本身是独立存储的。
为什么修改对象属性生效,但重新赋值不生效
关键在操作对象的哪一层:
- 执行
user.setName("Alice"):通过副本地址找到原对象并修改其字段 → 主调方可见变化 - 执行
user = new User():只是把形参变量指向了新对象,原实参变量仍指向旧对象 → 主调方无感知
示例:
void modify(User u) {
u.setName("Bob"); // ✅ 生效:改的是堆中同一个对象
u = new User(); // ❌ 无效:u 现在指向新对象,和外部 u 无关
}
基本类型和引用类型的传递表现对比
两者都是值传递,但“值”的含义不同:
立即学习“Java免费学习笔记(深入)”;
-
int x = 5→ 传的是数字5的副本,改形参不影响实参 -
User u = new User()→ 传的是类似0x7a8b这样的内存地址副本,两个变量初始都存这个地址
因此,String、Integer 等不可变类也遵循同样规则:你无法通过形参改变实参所指对象的内容(因为不可变),更无法让实参指向别处(因为赋值只改形参)。
容易踩坑的典型场景
这些地方常因混淆“对象可变性”与“传递机制”而出错:
- 在方法内用
list = new ArrayList()替换集合 → 外部 list 不受影响 - 期望通过
swap(User a, User b)交换两个引用变量 → Java做不到,C++才支持引用传参 - 误以为
StringBuilder.append()是“改变了引用” → 实际是修改了堆中对象状态,引用地址没变
真正需要“传出多个值”时,得靠返回对象、封装容器或使用数组/AtomicReference等间接方式。









