Arrays.asList 返回的 List 不可修改,底层为固定大小的 Arrays$ArrayList,调用 add/remove 抛 UnsupportedOperationException;需可变列表应显式拷贝为 new ArrayList(Arrays.asList(...)) 或使用 Arrays.stream().collect(Collectors.toList())。

Arrays.asList 不能直接修改枚举 List
用 Arrays.asList 转枚举数组,得到的是固定大小的 List,底层是 Arrays$ArrayList(非 java.util.ArrayList),不支持 add、remove 等操作,调用会抛 UnsupportedOperationException。
- 适用场景:只读遍历、传参给需要
List接口的方法(如Collections.unmodifiableList包装前) - 注意:如果原始数组后续被修改,
Arrays.asList返回的 List 会同步反映变化(共享底层数组) - 示例:
Arrays.asList(Color.RED, Color.GREEN)返回不可变视图,不是副本
EnumSet 是更轻量且线程不安全的枚举专用集合
EnumSet 是专为枚举设计的高性能集合,底层用位向量实现,内存占用极小,迭代快,但不保留插入顺序(按枚举声明顺序排列),且所有实例都非线程安全。
- 创建必须用静态工厂方法:
EnumSet.of、EnumSet.range、EnumSet.copyOf;不能用new -
EnumSet.copyOf(enumArray)是最常用转换方式,返回可变集合,支持增删 - 不兼容非枚举类型,传入
null或空数组会抛NullPointerException或IllegalArgumentException - 性能上比
new ArrayList(Arrays.asList(...))快 2–3 倍,尤其在元素少、频繁迭代时明显
要可变 List 就得显式拷贝
如果业务逻辑明确需要一个「能增删、独立于原数组、保留顺序」的 List,就得绕过 Arrays.asList 的陷阱,手动构造。
- 推荐写法:
new ArrayList(Arrays.asList(myEnumArray))—— 先视图再拷贝 - 等价但更直白:
Arrays.stream(myEnumArray).collect(Collectors.toList())(Java 8+) - 避免写成
new ArrayList(EnumSet.copyOf(myEnumArray)):虽然可行,但多一次遍历,且顺序由枚举定义顺序决定,未必符合数组原有顺序 - 如果数组可能为
null,务必先判空,Arrays.asList(null)不报错,但后续遍历时可能触发 NPE
EnumSet 和 ArrayList 在序列化/泛型推导上的差异
两者在泛型擦除和序列化行为上表现不同,容易在 RPC 或持久化场景中踩坑。
立即学习“Java免费学习笔记(深入)”;
-
EnumSet是抽象类,反序列化时需确保运行时有对应具体子类(如RegularEnumSet或JumboEnumSet),否则可能失败 -
ArrayList序列化稳定,但泛型信息在运行时丢失;若用ObjectMapper反序列化 JSON,需显式传TypeReference,否则默认转成LinkedHashMap - 编译期泛型推导:
EnumSet.of(Color.RED)推出EnumSet<color></color>;而Arrays.asList(Color.RED)推出List<color></color>,类型更通用但无枚举语义保障
EnumSet;要插元素、保持数组顺序、交其他框架(比如 MyBatis 参数绑定)→ 老老实实 new 个 ArrayList。别指望 Arrays.asList 给你一个“看起来像 List 实际能改”的东西。










