Java所有参数都是值传递:基本类型传数值副本,引用类型传地址副本;方法内重赋值不影响外部,但修改对象状态可见,因两者指向同一堆内存。

Java里所有参数都是值传递
Java没有真正的“引用传递”,连对象也是值传递——只是这个“值”是对象引用的副本。这意味着:方法内部对形参变量的重新赋值(比如 obj = new Object())不会影响外部实参;但通过该引用修改对象状态(比如 obj.setName("xxx")),外部可见,因为两个引用指向同一块堆内存。
基本类型参数传递:纯值拷贝
传的是变量本身的数值副本,方法内任何修改都与原变量无关。
public static void changeInt(int x) {
x = 100;
}
int a = 5;
changeInt(a);
System.out.println(a); // 输出 5,未变
-
int、boolean、char等八种基本类型,传参时直接复制栈上存储的值 - 方法内改
x,只改了栈帧里的局部变量,不影响调用方的a - 不存在“通过参数反向修改原始变量”的可能
引用类型参数传递:引用的值被拷贝
传的是对象引用(即内存地址)的副本,不是对象本身,也不是引用的引用。
public static void changeList(Listlist) { list.add("new"); // ✅ 修改对象状态 → 外部可见 list = new ArrayList<>(); // ❌ 重赋值引用 → 不影响外部 list 变量 } List names = new ArrayList<>(); names.add("old"); changeList(names); System.out.println(names); // [old, new],不是空列表
- 方法内
list.add(...)操作的是堆中原始对象,所以生效 - 但
list = new ArrayList()只让形参指向新对象,原变量names仍指向旧对象 - 数组、自定义类、
StringBuffer同理;但注意String是不可变的,所有“修改”实际是新建对象
容易混淆的点:String 和包装类的“假引用行为”
它们是引用类型,但因不可变性,表现像值类型——每次“修改”都会生成新对象,原引用不变。
立即学习“Java免费学习笔记(深入)”;
public static void changeString(String s) {
s = s + " world"; // 创建新 String 对象,s 指向它
}
String msg = "hello";
changeString(msg);
System.out.println(msg); // 输出 "hello",没变
-
String、Integer、Double等包装类,内部状态不可变 - 看似“修改”实为重新赋值,且新对象仅绑定到形参变量,不波及外部
- 误以为这是“值传递基本类型”的表现,其实是“引用传递 + 不可变对象”的组合效果
String 参数能被方法内拼接操作改变原值。










