当前位置: 首页 > 知识库问答 >
问题:

java多线程-限制提交到ExecutorService

党源
2023-03-14

我有一个包含数千行的数据文件。我正在读取它们并将它们保存在数据库中。我想以50行的批次多线程处理这个过程。当我在文件中读取时,10行被提交给ExecutorService。

ExecutorService executor = Executors.newFixedThreadPool(5);`

我可以在一段时间内循环执行以下操作,直到行结束。。。。

 Future<Integer> future = executor.submit(callableObjectThatSaves10RowsAtOneTime);

但是,如果处理10行需要时间,我不想将整个文件读入内存。我只想在其中一个线程返回之前提交5个线程,然后提交下一个线程。

假设一个线程需要20秒来保存10条记录,我不希望ExecutorService被输入数千行,因为读取过程正在继续读取并提交给ExecutorService

实现这一目标的最佳方法是什么?

共有1个答案

漆雕欣德
2023-03-14

您可以使用链接列表来执行此操作

int threads = 5;
ExecutorService service = Executors.newFixedThreadPool(threads);
LinkedList<Future<?>> futures = new LinkedList<>();

//As long as there are rows to save:
while(moreRowsLeft()){
    //dump another callable onto the queue:
    futures.addLast(service.submit(new RowSavingCallable());

    //if the queue is "full", wait for the next one to finish before
    //reading any more of the file:
    while(futures.size() >= 2*threads) futures.removeFirst().get();
}

//All rows have been submitted but some may still be writing to the DB:
for(Future<?> f : futures) future.get();

//All rows have been saved at this point

您可能想知道为什么我允许未来的数量达到机器上线程数量的两倍-这允许执行器服务线程处理数据库保存,而主线程正在创建更多的工作要做。这有助于在工作线程忙于数据库写入时隐藏与提供更多可调用项以供处理相关的任何I/O开销。

 类似资料:
  • 所以我们有一些游戏服务器在我们的V服务器上运行(我说的这款有8Vcore、4.2GHz和32GB DDR4内存),例如Minecraft。我们的问题是服务器在大约640个线程的情况下耗尽了内存(它不关心栈大小,1024KB、512KB都是相同的结果)。那么有没有办法从我们的系统中得到更多的线程呢? Linux:Debian 9 Virtuozzo容器 Java: openjdk版本“1.8.023

  • 问题内容: 我在实践中阅读Java Concurrency,并且有点与线程限制概念混淆。这本书说 当一个对象被限制在一个线程中时,即使该限制对象本身不是一个线程,这种使用也是自动的线程安全的 那么,当一个对象被限制在一个线程中时,没有其他线程可以访问它吗?那就是局限于线程吗?如何将对象限制在线程中? 编辑: 但是,如果我仍然想与另一个线程共享对象怎么办?假设在线程A完成对象O后,线程B想要访问O。

  • 我需要做一些多线程工作,并使用ExecutorService。newCachedThreadPool()并提交从队列中检索到的一些作业。 所以每个线程都创建一个的新实例,我在其中做一些工作。 我从另一个类调用我的,所以在提交所有作业后,我这样做: 但它只会杀死我的线程,即使工作没有完成,也没有等待。 也许是因为我在每个线程中创建了一个新的实例?

  • 我正在尝试创建一个具有一定数量线程的ThreadPoolExector,但同时,我想控制池队列的大小。所以我使用完整的构造函数创建了执行器: 然而,这给了我一个非法辩论例外。如果我将构造函数更改为 它起作用了。如果我希望理想的线程数和最大线程数相同,为什么它不起作用呢。

  • 问题内容: 基本上我需要在更多线程中运行〜数百个计算。我只想在paralell中运行一些并行线程,例如5个线程和5个计算。 我正在使用spring框架,@Async选项是自然选择。我不需要全功能的JMS队列,这对我来说有点麻烦。 有任何想法吗 ?谢谢 问题答案: 你检查了吗?你可以定义一个线程池,其中包含最大数量的线程来执行任务。 如果要与结合使用,请在spring-config中使用它:

  • 我有一个应用程序,允许用户批量向图像添加水印。该应用程序将只使用一个线程,并且一次只能添加一个水印。 我希望用户能够更改一次运行的水印任务[线程]的数量:可能在设置中为[1-5],并且我不能使用固定的ThreadPool,因为它具有固定的池大小。 我研究了如何使用线程池执行器(ThreadPoolExecutor)私有静态线程池执行器(ThreadPoolExecutor)=(ThreadPool