
循环里取数组元素总报 ArrayIndexOutOfBoundsException?先看索引算对没
绝大多数 ArrayIndexOutOfBoundsException 不是数组真越界,而是循环变量和数组长度关系没理清。Java 数组下标从 0 开始,最大合法索引是 array.length - 1,但很多人写 for (int i = 0; i —— 这个 <code> 就是雷。
常见错误现象:
- 遍历长度为 5 的数组,
i跑到 5 才停,访问array[5]立刻崩 - 倒序循环写成
for (int i = array.length; i >= 0; i--),第一轮就访问array[array.length] - 嵌套循环里复用外层索引变量,内层没重置或范围错配
实操建议:
- 正向遍历统一用
for (int i = 0; i ,死记“小于,不是小于等于” - 倒序遍历用
for (int i = array.length - 1; i >= 0; i--),起点必须减一 - 如果用增强 for(
for (Type e : array)),彻底绕开索引——但前提是不需要知道当前下标
多维数组嵌套循环的边界怎么同步校验
二维数组 matrix[i][j] 的两个维度长度可能不同,matrix.length 是行数,matrix[i].length 才是第 i 行的列数。拿错长度就会在某一行突然越界。
立即学习“Java免费学习笔记(深入)”;
常见错误现象:
- 假设所有行等长,用
matrix[0].length当作每行长度,但某行是null或长度更短 - 外层循环用
i ,内层却写 <code>j (把行数当列数)
实操建议:
- 内层循环前加空值检查:
if (matrix[i] != null),再用matrix[i].length - 不要硬编码
matrix[0].length,除非你 100% 确认矩阵规整且非空 - 用
Arrays.stream(matrix).allMatch(row -> row != null && row.length == expected)做预检(仅调试期)
循环中动态修改数组/集合长度导致的隐性越界
数组本身不可变长,但很多人误以为在循环里 list.remove(i) 或 list.add() 后,原数组引用还“保活”。实际是:操作的是集合,但循环仍按旧长度跑,或者迭代器失效后继续调用 next()。
常见错误现象:
- 用普通 for 遍历
ArrayList,中间remove(i),后面元素前移,但i还自增,跳过下一个元素,后续可能访问到null或越界 - 用增强 for 遍历集合时调用
remove(),直接抛ConcurrentModificationException(虽不是ArrayIndexOutOfBoundsException,但同属边界失控)
实操建议:
- 要边遍历边删,用
Iterator.remove(),别用索引删 - 真要用索引删,循环用
while (i ,删完不 <code>i++(因为下个元素已移到当前位置) - 避免在循环体里改变被遍历对象的结构——这是最省心的预防
边界逻辑藏在表达式里:subarray、substring、Arrays.copyOfRange 的坑
这些方法不直接用 for,但底层仍是数组索引运算。比如 str.substring(3, 3) 合法(返回空串),但 str.substring(3, 4) 在 str.length() == 3 时就崩;Arrays.copyOfRange(arr, 1, 5) 要求 5 ,不是 <code>。
实操建议:
- 所有带两个索引参数的方法,都遵守“左闭右开”:起始 ≥ 0,结束 ≤ 数组/字符串长度,且起始 ≤ 结束
- 调用前手动校验:
if (from >= 0 && to ,尤其当 from/to 来自用户输入或计算结果时 - 用
Math.min(to, arr.length)截断结束位置——适合容错场景,但得确认业务是否允许截断
边界逻辑最麻烦的地方不在循环语法,而在“谁负责校验”。JVM 不替你判断 i + 2 是否超限,也不会提醒你 list.get(i + 1) 在最后一轮会越界。每次写带索引的访问,都要默念一遍上下界——这没法偷懒。










