Java中List出现IndexOutOfBoundsException的根本原因是试图访问一个不存在的索引位置,即下标为负数或大于等于list.size();有效索引范围是0到size()-1,常见错误包括get(size())、get(-1)及循环越界。

Java中List出现IndexOutOfBoundsException,根本原因是**试图访问一个不存在的索引位置**——要么下标为负数,要么大于等于当前List的实际大小(size())。
索引从0开始,但size()返回的是元素个数
这是最容易忽略的细节。比如一个List有3个元素,它的有效索引是0、1、2,而list.size()返回的是3。如果写list.get(3),就会越界。
-
list.get(list.size())→ 总是越界(最大合法索引是size()-1) -
list.get(-1)→ 总是越界(最小合法索引是0) - 循环里用
i 代替i → 常见笔误
遍历时修改了List结构
在for循环或增强for(foreach)中,一边遍历一边调用remove()或add(),会改变List长度和元素位置,导致后续索引错位或访问到空位。
- 例如:遍历
[a,b,c],在i=1时删掉b,列表变成[a,c],但循环继续取i=2 → 越界 - 增强for底层用Iterator,此时会抛
ConcurrentModificationException,但某些并发场景或手动操作可能绕过检查,间接引发越界 - 安全做法:用迭代器的
remove()方法,或倒序for循环删除
多线程环境下未同步访问
多个线程同时读写同一个ArrayList(非线程安全),可能一个线程刚判断i 为true,另一个线程立刻清空了List,导致执行get(i)时实际size已为0。
立即学习“Java免费学习笔记(深入)”;
- 这不是“代码写错了”,而是竞态条件引发的逻辑越界
- 解决方式:加锁、用
Collections.synchronizedList()、或改用CopyOnWriteArrayList(适合读多写少) - 注意:
synchronizedList只保证单个方法原子性,if (i 这种两步操作仍需额外同步
初始化或赋值后没校验是否为空
尤其在从外部获取List(如方法返回、JSON解析、数据库查询)时,容易假设它“一定有数据”,直接取get(0)。
- 返回null?先判空;返回空List?再判
isEmpty()或size()>0 -
工具类如Apache Commons Collections提供
CollectionUtils.isNotEmpty(list),更简洁 - Optional包装返回值、或使用Stream的
findFirst(),也能避免显式索引操作
基本上就这些。越界不是语法错误,而是运行时逻辑疏漏——盯住索引合法性、避开遍历中修改、注意并发、别盲目信任输入。不复杂但容易忽略。










