Java中二维数组实为“数组的数组”,声明推荐int[][] arr,支持静态初始化如{{1,2},{3}}和动态初始化如new int3或逐行new;未初始化第二维会导致NullPointerException。

Java中二维数组的声明和初始化方式
Java里没有真正意义上的“二维数组”,只有“数组的数组”。这意味着第一维是一个一维数组引用,而每个元素又指向另一个一维数组(长度可不同)。声明时必须明确第一维大小,第二维可在运行时分别分配。
-
int[][] arr是推荐写法,强调类型是“二维整型数组引用”;int[] arr[]语法合法但易混淆,不建议 - 声明不分配内存:
int[][] matrix;此时matrix为null - 静态初始化(编译期确定):
int[][] grid = { {1, 2, 3}, {4, 5}, {6, 7, 8, 9} };每行长度可以不同,形成“锯齿数组”(jagged array) - 动态初始化(运行时分配):
int[][] table = new int[3][4]; // 所有子数组长度固定为4 int[][] ragged = new int[3][]; // 第二维未指定,必须后续逐个 new ragged[0] = new int[2]; ragged[1] = new int[5]; ragged[2] = new int[1];
访问二维数组元素时的常见 NullPointerException
因为第二维是独立对象,若只声明了第一维、未对某行执行 new,访问该行任意列就会抛出 NullPointerException。
- 错误示例:
int[][] data = new int[2][]; System.out.println(data[0][0]); // 运行时报错:java.lang.NullPointerException
- 正确做法:访问前检查
data[i] != null,或确保每行都已初始化 - 遍历时推荐用增强 for 循环避免越界,但要注意内层循环仍需判空:
for (int[] row : data) { if (row == null) continue; for (int val : row) { System.out.print(val + " "); } }
二维数组与嵌套 ArrayList 的性能和灵活性对比
当需要频繁增删行/列,或行列长度变化剧烈时,ArrayList 更灵活;但若数据稳定、访问密集,原生二维数组更快且内存更紧凑。
- 二维数组优势:连续内存布局、无装箱开销(基本类型)、随机访问 O(1)
- 嵌套
ArrayList优势:支持动态扩容、可调用add()/remove()、天然支持 null 元素 - 混合使用场景:用
int[][]存原始数据,用List管理行集合(避免泛型擦除带来的装箱)










