
本文详解在 java 中正确移除 arraylist 中 null 元素的多种方法,重点剖析正向遍历删除导致漏删的根本原因,并提供反向遍历、迭代器、stream api 等专业级解决方案,附可运行示例与关键注意事项。
在 Java 开发中,尝试从 ArrayList
例如,原始列表为 [A, null, null, B],当 i == 1 时移除第一个 null,列表变为 [A, null, B],此时原索引 2 处的 null 移至索引 1;但下一轮 i 已增至 2,于是该 null 被完全忽略——这正是提问者观察到“null 依然存在”的技术根源。
✅ 推荐方案一:反向遍历(最简洁、零依赖)
通过从末尾向前遍历,避免索引偏移干扰,代码简洁且性能优异:
for (int i = words.size() - 1; i >= 0; i--) {
if (words.get(i) == null) {
words.remove(i);
}
}? 提示:也可写作 for (int i = words.size(); --i >= 0; )(前置自减),语义等价,部分开发者认为更紧凑。
✅ 推荐方案二:使用 Iterator(符合集合操作规范)
Iterator.remove() 是唯一安全的遍历中删除方式,语义清晰且线程安全(单线程场景):
立即学习“Java免费学习笔记(深入)”;
Iteratorit = words.iterator(); while (it.hasNext()) { if (it.next() == null) { it.remove(); // 安全删除,无 ConcurrentModificationException 风险 } }
✅ 推荐方案三:Java 8+ Stream API(函数式、不可变风格)
若需保持原列表不变并生成新列表,推荐使用 Stream.filter():
words = words.stream()
.filter(Objects::nonNull) // 移除所有 null
.collect(Collectors.toCollection(ArrayList::new));⚠️ 注意:此方式创建新列表,原引用被替换;若需就地修改且兼容旧版本 JDK,优先选方案一或二。
? 补充说明:为何示例输出含空字符串 ""?
提问者示例输出中出现 [, , , ](即空字符串),而非 null。需明确区分:
- null 是引用为空,== null 判断成立;
- "" 是合法字符串对象,需用 .isEmpty() 或 .isBlank()(Java 11+)判断。
如需同时清理 null 和空字符串,可组合过滤:
words.removeIf(s -> s == null || s.isEmpty()); // Java 8+
// 或 Stream 方式:
words = words.stream()
.filter(s -> s != null && !s.isEmpty())
.collect(Collectors.toCollection(ArrayList::new));✅ 最佳实践总结
- 永远避免正向 for 循环 + list.remove() —— 这是初学者高频陷阱;
- 优先使用 removeIf()(Java 8+):语义最直观,一行解决,内部已优化;
- 处理大规模数据时注意性能:ArrayList.remove(i) 时间复杂度为 O(n),频繁删除建议先收集索引再批量移除,或改用 LinkedList(仅当插入/删除远多于随机访问时);
- 调试技巧:打印 words.size() 和每轮 i 值,快速验证索引逻辑是否异常。
通过以上任一正确方案,即可稳定获得期望结果:[10, years, 3, Lec, 3, Lab, Coordinating, Board, Academic, Approval, Number, 1102015707]。










