java – 在循环中将Runnable提交给线程池是一种好习惯吗?

我有以下代码片段,我想知道在线程池上运行任务时是否在循环中提交Runnable实例是一种很好的做法.

我需要访问循环外的列表,这是我的推理.这是伪代码,所以我的真实代码使用ConcurrentHashMap,消除了线程问题.如果这是不好的做法,有没有人有更好的建议?我尝试将其拆分为另一个类,但遇到了我的外部列表问题.

我知道何时清除内存列表时遇到麻烦,我无法知道线程何时全部完成.

public void startJob() {
    int threads = Runtime.getRuntime().availableProcessors();
    ExecutorService exec = Executors.newFixedThreadPool(threads);

    final List<ImportTask> importTasks = session.createCriteria(ImportTask.class).list();
    final List<Object> objs = new ArrayList<>();

    int count = 0;

    for (ImportTask importTask : importTasks) {
        exec.submit(new Runnable() {
            @Override
            public void run() {
                count++;

                if(objs.contains(importTask) {
                    obj = objs.get(importTask.indexOf(importTask));
                } else {
                    Object obj = new Object();
                    objs.add(obj);
                    session.save(obj);
                }


                if(count % 50 = 1000) {
                    session.flush();
                    session.commit();
                }
            }
        }
    }
}

解决方法:

将Runnable实例提交到线程池是实现任务并发执行的完美有效方法.

要知道每个任务何时完成,请保留ExecutionService.submit()返回的Future对象列表.

Future<?> future = exec.submit(new Runnable() {
...
futures.add(future);

最后使用阻塞Future.get()获取每个未来的结果.

正如另一个答案所示,您也可以使用ExecutorService.invokeAll().不同之处在于该方法仅采用Callable而不是Runnable,必须一次性提交所有任务,并且该方法等待直到最后一个任务终止.

使代码更整洁的另一个步骤是使您的任务(ImportTask)实现Callable接口(和调用方法)并返回它的计算.如果需要将值传递给任务,例如objs列表,则可以通过ImportTask的构造函数执行此操作.

上一篇:java线程同步 – 这不应该工作,但它是:) –


下一篇:c# – 我应该在自己的线程中运行每个插件吗?