Java数组是固定长度、同类型、连续内存的引用类型对象;声明int[] arr仅创建null引用,需new或初始化器才真正分配堆内存,索引从0开始因内存寻址公式为“首地址+index×元素字节数”。

Java里的数组是一个**固定长度、同类型、连续内存**的引用类型对象,不是语法糖,也不是基础类型——它在堆上分配,变量里存的是地址。
数组声明和创建为什么必须分开理解?
很多人写 int[] arr; 就以为“有了数组”,其实这只是声明了一个引用变量,值为 null。真正创建数组必须用 new 或初始化器,否则后续访问会直接抛 NullPointerException。
-
int[] arr;→ 引用未指向任何对象,arr.length会报空指针 -
int[] arr = new int[5];→ 分配了 5 个int的连续空间,每个默认为0 -
int[] arr = {1, 2, 3};→ 静态初始化,等价于new int[]{1,2,3},长度由元素个数决定
为什么索引从 0 开始,且最大下标是 length - 1?
这不是 Java 特色,而是底层内存寻址逻辑决定的:数组首地址 + index × 单元素字节数。越界访问(比如对长度为 3 的数组读 arr[3])会触发 ArrayIndexOutOfBoundsException,JVM 不做容错。
- 常见误判:
for (int i = 1; i → 必然越界 - 安全写法:
for (int i = 0; i ,永远用,不用 - 增强 for(
for (int x : arr))能避免越界,但拿不到下标,改值也无效(只是副本)
Arrays 工具类哪些方法真该日常用?
别手写打印、拷贝、比较——java.util.Arrays 提供了经过充分测试的高效实现,且语义清晰。
立即学习“Java免费学习笔记(深入)”;
- 打印内容:
System.out.println(Arrays.toString(arr)),比循环输出快、可读性强 - 深拷贝(新数组):
int[] copy = Arrays.copyOf(arr, arr.length);想扩容就传更大长度,如Arrays.copyOf(arr, arr.length * 2) - 判断相等:
Arrays.equals(a, b)比a == b(地址比较)或a.equals(b)(默认引用比较)靠谱得多 - 注意:
Arrays.asList(arr)返回的不是ArrayList,而是不可变的固定大小列表,add()会抛UnsupportedOperationException
数组一旦创建,长度就锁死——这是它和 ArrayList 最根本的区别。如果业务需要频繁增删,数组不是第一选择;但如果明确知道数据量、追求极致遍历性能或与 JNI/IO 等底层交互,数组仍是不可替代的基石。别把它当“简化版集合”用,而要理解它作为连续内存块的本质约束。










