int[]声明只创建栈中引用变量,值为null;new int[5]或{1,2,3}才在堆中分配内存;数组长度不可变,“扩容”实为新建+复制。
Java里数组不是变量的别名,而是一个独立的对象引用——你操作的是地址,不是值本身。
int[] 声明到底创
写 int[] arr 只是声明了一个引用变量,此时 arr 的值为 null,没指向任何对象;必须用 new int[5] 或字面量 {1,2,3} 才真正分配堆内存。
int[] a; → a 是栈上一个 4/8 字节的引用槽,内容为 null
int[] b = new int[3]; → 堆中分配 3 个 int 连续空间,b 存的是该块起始地址int[] c = b; → 不复制元素,只是把 b 的地址值拷贝给 c,二者指向同一块内存对 arr[i] = 10 是修改堆中数据;对 arr = new int[10] 是让引用指向新地址——前者影响所有共享该引用的变量,后者只改变当前变量指向。
a[0] = 99; 后,若 b == a,则 b[0] 也变成 99
a = new int[]{5,6}; 后,b 仍指向原来的数组,不受影响int[] 存的是真实 int 值(堆中连续存储);Integer[] 存的是 Integer 对象引用,每个元素都可能为 null,且对象分散在堆中。
int[] x = {1,2}; → 内存紧凑,无装箱开销,x[0] 直接是二进制整数Integer[] y = {1,2}; → 实际调用 Integer.valueOf(1),生成两个独立对象,y[0] 是指向它的引用Arrays.equals(x, y) 编译失败:类型不兼容,不能直接比较int[] nums = {1, 2, 3};
Integer[] boxed = Arrays.stream(nums).boxed().toArray(Integer[]::new);
// 注意:nums 和 boxed 完全无关,修改一个不影响另一个
最易被忽略的是:数组长度一旦确定就不可变,所谓“扩容”本质是新建数组 + 复制 —— ArrayList 就是靠这个机制实现的。别指望 arr.length++ 能生效。