在Java里方法参数传递机制是什么_Java值传递与引用传递说明

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,未变
  • intbooleanchar 等八种基本类型,传参时直接复制栈上存储的值
  • 方法内改 x,只改了栈帧里的局部变量,不影响调用方的 a
  • 不存在“通过参数反向修改原始变量”的可能

引用类型参数传递:引用的值被拷贝

传的是对象引用(即内存地址)的副本,不是对象本身,也不是引用的引用。

public static void changeList(List list) {
    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 和包装类的“假引用行为”

它们是引用类型,但因不可变性,表现像值类型——每次“修改”都会生成新对象,原引用不变。

public static void changeString(String s) {
    s = s + " world"; // 创建新 String 对象,s 指向它
}
String msg = "hello";
changeString(msg);
System.out.println(msg); // 输出 "hello",没变
  • StringIntegerDouble 等包装类,内部状态不可变
  • 看似“修改”实为重新赋值,且新对象仅绑定到形参变量,不波及外部
  • 误以为这是“值传递基本类型”的表现,其实是“引用传递 + 不可变对象”的组合效果
Java 的参数机制统一而严格:只有值传递。所谓“对象能被修改”,是因为你拿到的是指向堆对象的地址副本,而不是因为语言支持引用传递。真正容易出错的地方,往往发生在试图通过参数交换两个引用,或误以为 String 参数能被方法内拼接操作改变原值。