
本文探讨了在Java中如何高效异步处理大型列表的分批操作,特别是当操作之间存在依赖关系时。针对传统列表分区可能导致的内存开销问题,文章提出并详细阐述了两种优化方案:一是利用List.subList()创建列表视图结合CompletableFuture进行精细控制,以减少内存占用;二是使用Java 8并行流作为更简洁的替代方案,实现高效的并发批处理。通过示例代码和注意事项,旨在帮助开发者在处理大规模数据时平衡性能与资源消耗。
挑战:大型列表的异步分批处理与内存考量
在现代Java应用中,处理包含成千上万甚至数十万个元素的大型列表是常见需求。当这些元素需要执行一系列依赖性操作,并且希望以异步方式提高处理效率时,CompletableFuture是一个强大的工具。常见的做法是将原始大列表分割成若干子列表,然后将每个子列表提交给一个CompletableFuture任务进行处理。
然而,这种“分区”操作如果实现不当,可能会引入显著的内存开销。例如,如果分区逻辑通过复制元素来创建新的子列表,那么一个拥有十万元素的列表被分成一千个包含一百元素的子列表时,内存中将同时存在一千个额外的List对象及其内部的元素引用,这可能导致内存占用激增,尤其是在分区粒度较细时。
以下是一个可能导致内存问题的传统分区方法示例:
存了个图
视频图片解析/字幕/剪辑,视频高清保存/图片源图提取
17 查看详情
import java.util.List;import java.util.concurrent.CompletableFuture;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.ArrayList; // 假设 firstOperation 返回 ArrayList// 假设 SomeObject 是一个简单的POJOclass SomeObject { String id; public SomeObject(String id) { this.id = id; } @Override public String toString() { return "SomeObject{" + "id='" + id + ''' + '}'; }}public class TraditionalBatchProcessing { // 假设 firstOperation 返回 List public static List firstOperation(List subList){ // 执行第一个操作,例如处理 SomeObject 并返回其ID return subList.stream().map(o -> o.id + "_processed_trad").collect(ArrayList::new, ArrayList::add, ArrayList::addAll); } public static void secondOperationWithFirstOpResult(List firstOpProducedList) { // 执行第二个依赖于第一个操作结果的操作 // 例如,打印结果或写入数据库 // firstOpProducedList.forEach(System.out::println); } public static void main(String[] args) { List someObjectList = new ArrayList(); for (int i = 0; i < 100000; i++) { // 10万条记录 someObjectList.add(new SomeObject("item_" + i)); } int partitionSize = 100; // 每个批次的大小 ExecutorService executorService = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors()); // 假设 ListUtils.partition 是一个外部库方法,并且它通过复制创建子列表 // 实际上 ListUtils.partition 通常会使用 subList,这里仅为说明内存问题 List<List> partitionList = new ArrayList(); for (int i = 0; i < someObjectList.size(); i += partitionSize) { int end = Math.min(i + partitionSize, someObjectList.size()); partitionList.add(new ArrayList(someObjectList.subList(i, end))); // 模拟复制创建子列表 } System.out.println("Created " + partitionList.size() + " sublists by copying."); partitionList.forEach(subList -> { CompletableFuture.supplyAsync(() -> firstOperation(subList), executorService) .thenAcceptAsync(TraditionalBatchProcessing::secondOperationWithFirstOpResult, executorService); }); // 实际应用中需要更严谨的CompletableFuture组合等待机制 executorService.shutdown(); // try { executorService.awaitTermination(1, java.util.concurrent.TimeUnit.MINUTES); } catch (InterruptedException e) { e.printStackTrace(); } }}
这种方法的核心问题在于`partitionList.add(new ArrayList(someObjectList.subList(i, end)))
立即学习“Java免费学习笔记(深入)”;
以上就是Java CompletableFuture与List分批处理:优化内存与性能的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/325438.html
微信扫一扫
支付宝扫一扫