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,二者指向同一块内存
修改数组元素 vs 修改数组引用
对 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,且对象分散在堆中。
立即学习“Java免费学习笔记(深入)”;
-
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++ 能生效。










