登录
首页 >  文章 >  java教程

ArrayList动态存储原理与应用解析

时间:2026-04-14 19:33:47 112浏览 收藏

ArrayList虽在不指定初始容量时也能正常工作,但频繁add会因1.5倍扩容和数组复制显著拖慢性能;尤其当业务数据量可预估(如单次请求返回50–200条订单)时,直接初始化为new ArrayList(300)等合理容量,能有效避免冗余扩容、提升运行效率——这看似微小的优化,实则是高并发、大数据量场景下保障系统响应速度的关键细节。

如何使用 ArrayList 存储动态增长的业务数据列表对象

ArrayList 初始化时要不要指定初始容量

多数情况下,不指定初始容量也能正常工作,但频繁 add 会导致多次扩容,触发内部数组复制,影响性能。尤其在业务数据量可预估时(比如单次请求通常返回 50–200 条订单),建议直接传入合理初始值。

扩容机制:默认容量是 10,每次扩容为原容量的 1.5 倍(JDK 17+),且需调用 Arrays.copyOf 复制整个数组。若你明确知道会存 300 个对象,初始化写成 new ArrayList(300) 更稳。

  • 小列表(
  • 中等批量(50–500):设初始容量能减少 1–2 次扩容
  • 高频写入循环内创建 ArrayList:务必指定,否则每轮都重分配

add() 时传 null 是否安全,业务对象字段为空怎么处理

ArrayList 允许存 null,但业务上多数场景应避免——它容易掩盖空指针或校验缺失问题。比如存 Order 对象时,若 order.getUserId() 返回 null,后续遍历时可能在别的地方抛 NullPointerException,而不是在入库前就暴露。

更稳妥的做法是在 add 前做轻量校验:

if (order != null && order.getUserId() != null) {
    list.add(order);
}
  • 数据库主键类字段(如 id、userId)为空,建议拒绝添加并记录 warn 日志
  • 非关键字段(如 remark)允许为空,不影响 add
  • 不要依赖 ArrayList 帮你过滤或容错——那是业务逻辑层该干的事

遍历 ArrayList 时修改内容会不会出错

直接用 for-each 或 list.iterator() 遍历时调用 list.remove()list.add(),会触发 ConcurrentModificationException。这不是线程问题,而是 fail-fast 机制在起作用。

正确做法取决于你要做什么:

  • 要边查边删:用 Iterator.remove(),例如 it.remove()
  • 要批量替换字段:直接按索引遍历 for (int i = 0; i ,然后 list.set(i, updated)
  • 要过滤后生成新列表:用 stream().filter().collect(Collectors.toList()),语义清晰且线程安全

注意:list.clear() 不会触发异常,但它只是清空引用,原对象还在堆里——如果其他地方还持有这些对象引用,它们不会被回收。

ArrayList 能不能直接当缓存用

不能。虽然它支持动态增删,但没过期机制、没容量淘汰、没线程安全包装。业务中常见错误是把 ArrayList 声明为 static 全局变量来“缓存用户列表”,结果并发写入时数据错乱,或者内存只增不减最终 OOM。

真要缓存,请换专用方案:

  • 简单时效缓存:用 Caffeine.newBuilder().maximumSize(1000).expireAfterWrite(10, TimeUnit.MINUTES)
  • 需分布式共享:走 Redis,用 LPUSH/LRANGE 模拟列表,但注意 Redis 列表不是 ArrayList 的替代品——它不支持随机更新
  • 纯本地、短生命周期临时聚合:ArrayList 可以,但必须是方法内局部变量,用完即弃

最容易被忽略的一点:ArrayList 的 size() 是 O(1),但 contains() 是 O(n),业务里如果高频调用 list.contains(obj) 查重,应该换成 HashSet 或加索引字段。

到这里,我们也就讲完了《ArrayList动态存储原理与应用解析》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>