Java中数组赋值为何会相互影响?深入理解引用传递机制

java中数组是引用类型,变量赋值只是复制引用而非创建新数组,因此修改副本会同步影响原数组。

在Java中,int[]看似与基本类型int相似,但本质完全不同:数组是对象,属于引用类型。这意味着声明 int[] arr = {1,2,3,4}; 实际上做了两件事:

  1. 在堆内存中创建一个包含4个整数的数组对象;
  2. 在栈中创建变量 arr,其值为该数组对象的内存地址(即引用)。

当执行 int[] y = arr; 时,并未创建新数组,而是将 arr 中存储的同一地址赋给了 y。此时

arr 和 y 指向堆中同一个数组对象:

arr ──────────┐
             ↓
         [1, 2, 3, 4]  ← 堆中的唯一数组对象
             ↑
y ────────────┘

因此,y[0] = 15; 实质是通过 y 找到该数组并修改其首元素——而 arr 指向的正是这个被修改的对象,所以 System.out.println(Arrays.toString(arr)); 输出 [15, 2, 3, 4] 完全符合预期。

✅ 正确的深拷贝方式(创建独立副本):

import java.util.Arrays;

public class Alle {
    public static void main(String[] args) {
        int[] arr = {1, 2, 3, 4};
        int[] y = Arrays.copyOf(arr, arr.length); // 创建新数组,内容相同但地址不同
        y[0] = 15;
        System.out.println(Arrays.toString(arr)); // 输出: [1, 2, 3, 4]
        System.out.println(Arrays.toString(y));   // 输出: [15, 2, 3, 4]
    }
}

⚠️ 注意事项:

  • y = arr.clone()、System.arraycopy() 或 new int[]{...} 也可实现独立拷贝;
  • y = Arrays.asList(arr) 不适用——它返回的是 List(包装了整个数组),且对原始数组仍无隔离;
  • 此机制适用于所有对象类型(如 String[], CustomClass[]),但不适用于基本类型变量(如 int a = b; 是值拷贝)。

理解“引用即地址”是掌握Java内存模型的关键——赋值操作从不自动复制对象,只复制指向它的指针。