Java中如何将List按照固定大小拆分为多组

答案:Java中拆分List常用方法有三种:1. 使用原生循环和subList,手动计算分组并创建新列表副本;2. 借助Guava库的Lists.partition,代码简洁但返回视图为原列表;3. 采用Java 8 Stream函数式编程,通过IntStream生成索引映射子列表。推荐小项目用Guava,无依赖需求则用手动循环,注意边界处理。

在Java中,可以将一个List按照固定大小拆分成多个子列表,这种操作常用于分批处理数据。虽然Java标准库没有提供直接的方法,但可以通过手动实现或借助第三方库来完成。

使用Java原生方法拆分List

通过循环和subList()方法,可以将List按指定大小分割:

  • 计算总共有多少组:(list.size() + batchSize - 1) / batchSize(向上取整)
  • 每轮使用subList(start, end)截取子列表
  • 注意subList返回的是原列表的视图,如需独立副本应使用new ArrayList(subList)

示例代码:

public static  List> partition(List list, int batchSize) {
    List> result = new ArrayList<>();
    for (int i = 0; i < list.size(); i += batchSize) {
        int end = Math.min(i + batchSize, list.size());
        result.add(new ArrayList<>(list.subList(i, end)));
    }
    return result;
}

调用方式:

List numbers = Arrays.asList(1,2,3,4,5,6,7,8,9,10);
List> groups = partition(numbers, 3);
// 每组最多3个元素

使用Guava工具库

Google Guava提供了更简洁的工具类Lists.partition:

List> partitions = Lists.partition(originalList, 3);
  • 需要引入Guava依赖(com.google.guava:guava)
  • 返回的子列表是原列表的视图,修改会影响原数据
  • 使用方便,代码简洁

使用Java 8 Stream实现(函数式风格)

通过IntStream生成索引并映射为子列表:

public static  List> partitionWithStream(List list, int size) {
    int total = list.size();
    return IntStream.iterate(0, i -> i + size)
                    .limit((total + size - 1) / size)
                    .mapToObj(i -> new ArrayList<>(list.subList(i, Math.min(total, i + size))))
                    .collect(Collectors.toList());
}
  • 适合喜欢函数式编程风格的场景
  • 可读性较好,但性能略低于传统循环

基本上就这些常用方式。小批量数据处理推荐使用Guava,避免重复造轮子;若不想引入外部依赖,手写循环最清晰可控。注意处理空列表和batchSize小于等于0的边界情况。不复杂但容易忽略细节。