17370845950

在Java中如何使用List.subList截取集合片段
subList返回原列表的视图,修改子列表会影响原列表,且原列表结构变化后操作子列表会抛ConcurrentModificationException,如需独立副本应使用new ArrayList(subList())。

在Java中,List.subList 方法可以用来获取集合的一个视图片段,而不是创建一个全新的独立列表。这意味着对子列表的操作可能会影响原列表,需要注意这一点。

subList的基本用法

调用 subList(int fromIndex, int toIndex) 可以截取从 fromIndex(包含)到 toIndex(不包含)之间的元素。

  • 参数 fromIndex:起始索引(从0开始)
  • 参数 toIndex:结束索引(不包含该位置元素)
  • 返回值是一个 List 视图,类型与原列表一致

示例代码:

List original = new ArrayList(); original.add("A"); original.add("B"); original.add("C"); original.add("D"); List part = original.subList(1, 3); System.out.println(part); // 输出 [B, C]

子列表与原列表的关联性

subList 返回的是原列表的视图,不是副本。因此:

  • 修改子列表中的元素会反映到原列表中
  • 如果原列表被结构化修改(如增删元素),再操作子列表会抛出 ConcurrentModificationException

例如:

List original = new ArrayList(Arrays.asList("X", "Y", "Z")); List view = original.subList(0, 2); view.set(0, "NEW"); // 修改子列表 System.out.println(original); // 输出 [NEW, Y, Z],原列表也被修改 original.add("W"); // 原列表结构改变 // view.add("test"); // 此时操作 view 会抛异常

创建独立副本避免影响

如果希望截取后的列表与原列表互不影响,应基于 subList 创建一个新的 ArrayList:

List safeCopy = new ArrayList(original.subList(1, 3));

这样得到的 safeCopy 是独立的,无论怎么修改都不会影响原列表,也避免了并发修改异常。

使用场景和注意事项

  • 适合用于临时处理集合某一段数据,节省内存(因为是视图)
  • 不要长期持有 subList 返回的对象,尤其是在原列表可能变动的情况下
  • 多线程环境下更要小心共享 subList 引发的问题
  • fromIndex 必须 ≤ toIndex,且两者都不能越界,否则抛 IndexOutOfBoundsException
基本上就这些。subList 使用简单但行为特殊,关键是理解它返回的是“视图”而非“复制”。