Java方法参数传递只有值传递:基本类型传值副本,修改不影响实参;对象类型传引用副本,可修改对象状态但不能改变原引用指向。

Java里方法参数传递只有值传递一种方式,所谓“引用传递”是常见误解——实际上传递的是对象引用的副本,不是引用本身。
基本类型参数传的是值的拷贝
传入 int、boolean、double 等基本类型时,方法内修改形参不会影响实参原始值。
public static void changeValue(int x) {
x = 100;
}
int a = 42;
changeValue(a);
System.out.println(a); // 输出 42,未变-
栈中为形参
x分配新空间,复制a的值过去 - 修改
x只影响这个副本,与a无关 - 这点和 C 完全一致,但和 Python 的“对象引用传递”语义不同
对象类型参数传的是引用的拷贝
传入 String、ArrayList、自定义类实例等时,传递的是堆中对象地址的副本,不是对象本身,也不是“引用的引用”。
public static void modifyList(ArrayListlist) { list.add("new"); // ✅ 能修改原对象内容 list = new ArrayList<>(); // ❌ 不会影响调用方的 list 变量 } ArrayList names = new ArrayList<>(); names.add("Alice"); modifyList(names); System.out.println(names.size()); // 输出 2
- 形参
list和实参names指向同一个堆对象,所以能修改其状态(如 add/remove) - 但若在方法内让
list指向新对象(= new ArrayList()),只是改了副本的指向,原变量names不受影响 -
String是特例:不可变,所有“修改”操作(如substring、toUpperCase)都返回新对象,原引用不变
数组参数的行为和对象一致
数组是对象,所以传的是数组引用的副本,可以修改元素,但不能让原数组变量指向新数组。
立即学习“Java免费学习笔记(深入)”;
public static void mutateArray(int[] arr) {
arr[0] = 999; // ✅ 修改成功
arr = new int[]{1,2,3}; // ❌ 调用方 arr 不变
}
int[] nums = {10, 20};
mutateArray(nums);
System.out.println(nums[0]); // 输出 999- 注意
arr.length是可读属性,不是方法调用,不涉及对象变更 - 如果需要“替换整个数组”,只能靠返回新数组 + 显式赋值:
nums = mutateArray(nums); - 泛型集合(
ArrayList等)没有这个问题,因为它们本身支持 clear/addAll 等就地操作
想真正“改变引用”该怎么办?
Java 语法不支持类似 C++ 的 & 引用参数或 C# 的 ref,但有几种实用替代方案:
- 返回新值并由调用方重新赋值:
list = filterAndSort(list); - 封装到可变容器里传入,例如
AtomicReference或自定义 holder 类- >
- 用数组包装单个值:
Integer[] holder = {value};,方法内改holder[0] - 避免强行模拟引用传递——多数时候说明设计可能需要重构:是否该把逻辑移到对象内部?是否该用 builder 模式?
最常被忽略的一点:很多人在调试时看到 IDE 显示“对象内容变了”,就以为是引用传递生效了,其实只是没意识到“修改对象状态”和“修改引用指向”是两回事。










