Executor/ExecutorService和ExecutorCompletionService

本文深入探讨Java SE 5引入的Executor框架,从Executor接口出发,介绍其与ExecutorService、AbstractExecutorService的关系,并详细解析Executors工具类及ExecutorCompletionService的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

上篇文章讲到了Callable和Futre,这篇文章接着来说说执行器相关的接口和类。在JavaSE5之后,在API的java.util.concurrent包中给出了Executor这样一个接口。因为在Java7之前,JDK中这个接口最终都是以一个线程池ThreadPool的方式实现的,所以很多人也直接叫“线程池”。除了ThreadPool之外,还有ForkJoinPool实现,本文中,我们不妨直接翻译Executor为执行器。

Executor本身只是个接口,但和这个概念相关有很多类:包括Executor本身、ExecutorService、AbstractExecutorService、提供诸多静态方法的工具类Executors和相关的ExecutorCompletionService类,下面会对这些接口和类一一整理。除此之外,还有刚刚提到的ThreadPoolExecutor和ForkJoinPool,这两个类后面我们再说。

0. 综述和类层次结构

JavaSE5之后的任务执行器Executor实际上取代了Java并发开发中直接使用Thread对象的操作,以Command Pattern(命令模式)的设计模式实现,成为Java并发开发的主角。

Executor是最为抽象的接口,ExecutorService继而扩展了Executor, AbstractExecutorService 则给出了抽象的实现。

1. Executor

先看java.util.concurrent.Executor这个接口:

public interface Executor {
    void execute(Runnable command);
}

同Runnable和Callable一样简洁,只有一个方法。注意execute()这个方法只允许有一个Runnable参数,也就意味着想用前文中的Callable,需要做适配工作。

2. ExecutorService

java.util.concurrent.ExecutorService也是一个接口,扩展了Executor,提供了更为丰富的方法。

public interface ExecutorService extendsExecutor {
    void shutdown();
    List shutdownNow();
    booleanisShutdown();
    boolean isTerminated();
    boolean awaitTermination(longtimeout, TimeUnit unit)
        throwsInterruptedException;
     Future submit(Callable task);
     Future submit(Runnable task, T result);
    Future<?> submit(Runnable task);
     List<Future> invokeAll(Collection<? extendsCallable> tasks)
        throwsInterruptedException;
     List<Future> invokeAll(Collection<? extendsCallable> tasks,
                                  longtimeout, TimeUnit unit)
        throwsInterruptedException;
     T invokeAny(Collection<? extendsCallable> tasks)
        throwsInterruptedException, ExecutionException;
     T invokeAny(Collection<? extendsCallable> tasks,
                    longtimeout, TimeUnit unit)
        throwsInterruptedException, ExecutionException, TimeoutException;
}

一共12个方法,其中一部分是和执行器生命周期相关的方法,而另一部分则是以各种方式提交要执行的任务的方法。像submit()就是提交任务的一个方法,在实现中做了适配的工作,无论参数是Runnable还是Callable,执行器都会正确执行。当然,这实际上用到的是前文提过的RunnableFuture的实现类FutureTask。

3. AbstractExecutorService

java.util.concurrent.AbstractExecutorService这个类是ExecutorService的一个抽象实现。其中,提交任务的各类方法已经给出了十分完整的实现。之所以抽象,是因为和执行器本身生命周期相关的方法,在此类中并未给出任何实现,需要子类扩展完善。

4. Executors
这个类同Arrays和Collections很类似,java.util.concurrent.Executors是个工具类,提供了很多静态的工具方法。其中很多对于执行器来说就是初始化构建用的工厂方法。其中部分方法名整理成如下列表:

  • 重载实现的newFixedThreadPool()
  • 重载实现的newSingleThreadExecutor()
  • 重载实现的newCachedThreadPool()
  • 重载实现的newSingleThreadScheduledExecutor()
  • 重载实现的newScheduledThreadPool()

这些方法返回的ExecutorService对象最终都是由ThreadPoolExecutor实现的,根据不同的需求以不同的参数配置,或经过其它类包装。其中,Executors中的一些内部类就是用来做包装用的。

Executors还提供了这样两个静态重载的方法:

publicstatic ExecutorService unconfigurableExecutorService(ExecutorService executor);
publicstatic ScheduledExecutorService unconfigurableScheduledExecutorService(ScheduledExecutorService executor);

这两个方法对参数中的executor做了包装,保证了Executor在初始化构造后不再可配置。

此外,Executors类中还有静态的defaultThreadFactory()方法,免去了自己构造ThreadFactory或是需要线程时对Thread类的new操作。

关于更多Executors类中工具方法的使用,还是建议去看API JavaDoc和JDK源码。

5. ExecutorCompletionService

前一篇文章提过了CompletionService这个接口,和执行器相关的,java.util.concurrent中提供了这个接口的一个实现,就是ExecutorCompletionService。

这个实现类主要做的就是将执行完成的任务结果放到阻塞队列中,这样等待结果的线程,如执行take()方法会得到结果并恢复执行。

ExecutorCompletionService有3个属性:

  • AbstractExecutorService类的对象aes
  • Executor类的对象executor
  • BlockingQueue<Future<V>>的completionQueue

通常,如果executor是AbstractExecutorService的一个实现,则将其赋值给aes属性,否则赋值为null。

在这个类中,executor负责执行任务,而aes则负责做适配处理,返回包装好任务的FutureTask对象。

这里面有一个对于实现功能很重要的内部类QueueingFuture,实现如下:

private class QueueingFuture extendsFutureTask {
    QueueingFuture(RunnableFuture task) {
        super(task,null);
        this.task = task;
    }
    protectedvoid done() { completionQueue.add(task); }
    privatefinal Future task;
}

主要是扩展了FutureTask的done方法,将执行结果放入BlockingQueue中。

### ExecutorService 等待子线程结束的原理及实现 #### 1. 线程池与 JVM 生命周期的关系 在 Java 中,主线程退出后,JVM 不会立即终止。只要存在非守护线程(Daemon Thread),JVM 就会继续运行[^4]。`ExecutorService` 创建的线程默认是非守护线程,因此即使主线程结束,线程池中的任务仍然会继续执行。如果主线程不等待子线程完成,可能会导致以下问题: - **资源泄漏**:未完成的任务可能持有对资源(如文件、数据库连接)的引用,导致资源无法释放。 - **数据丢失或不一致**:如果子线程正在执行关键操作(如写入日志或保存数据),主线程的提前退出可能导致数据丢失或不一致。 #### 2. `shutdown()` `awaitTermination()` 的作用 为了确保所有任务完成并优雅地关闭线程池,通常使用以下方法: - `shutdown()`:停止接收新任务,但允许已提交的任务继续执行[^3]。 - `awaitTermination(long timeout, TimeUnit unit)`:阻塞当前线程,直到线程池中的所有任务完成或达到指定超时时间[^4]。 以下是实现代码示例: ```java ExecutorService executorService = Executors.newFixedThreadPool(4); // 提交任务 for (int i = 0; i < 10; i++) { final int taskId = i; executorService.submit(() -> { System.out.println("Task " + taskId + " is running..."); try { Thread.sleep(1000); // 模拟任务耗时 } catch (InterruptedException e) { Thread.currentThread().interrupt(); } System.out.println("Task " + taskId + " completed."); }); } // 关闭线程池并等待任务完成 executorService.shutdown(); // 停止接收新任务 try { if (!executorService.awaitTermination(60, TimeUnit.SECONDS)) { // 等待任务完成或超时 executorService.shutdownNow(); // 强制关闭线程池 } } catch (InterruptedException e) { executorService.shutdownNow(); // 中断时强制关闭线程池 } ``` #### 3. 使用 `CountDownLatch` 实现等待 除了 `shutdown()` `awaitTermination()`,还可以使用 `CountDownLatch` 来等待所有子线程完成。`CountDownLatch` 是一个同步工具类,允许一个或多个线程等待其他线程完成操作[^2]。 以下是使用 `CountDownLatch` 的示例: ```java import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class CountDownLatchExample { public static void main(String[] args) throws InterruptedException { int numThreads = 5; CountDownLatch latch = new CountDownLatch(numThreads); ExecutorService executorService = Executors.newFixedThreadPool(numThreads); for (int i = 0; i < numThreads; i++) { final int taskId = i; executorService.submit(() -> { System.out.println("Task " + taskId + " started..."); try { Thread.sleep(1000); // 模拟任务耗时 } catch (InterruptedException e) { Thread.currentThread().interrupt(); } System.out.println("Task " + taskId + " completed."); latch.countDown(); // 任务完成计数减一 }); } latch.await(); // 主线程等待所有子线程完成 System.out.println("All tasks completed."); executorService.shutdown(); // 关闭线程池 } } ``` #### 4. `CompletionService` 的使用 `CompletionService` 是另一种管理任务完成的方式,它允许按任务完成顺序获取结果。通过结合 `ExecutorService` `Future`,可以更灵活地处理多线程任务的结果[^3]。 以下是 `CompletionService` 的示例: ```java import java.util.ArrayList; import java.util.List; import java.util.concurrent.*; public class CompletionServiceExample { public static void main(String[] args) throws Exception { ExecutorService executor = Executors.newFixedThreadPool(5); CompletionService<String> completionService = new ExecutorCompletionService<>(executor); // 提交任务 for (int i = 0; i < 10; i++) { final int taskId = i; completionService.submit(() -> { Thread.sleep((10 - taskId) * 100); // 模拟不同耗时 return "Task-" + taskId + " result"; }); } // 获取并合并结果(按完成顺序) List<String> results = new ArrayList<>(); for (int i = 0; i < 10; i++) { Future<String> future = completionService.take(); // 阻塞直到有任务完成 results.add(future.get()); // 获取任务结果 } executor.shutdown(); // 关闭线程池 System.out.println("Results in completion order: " + results); } } ``` #### 5. 原理总结 `ExecutorService` 等待子线程结束的核心原因在于线程池的生命周期管理线程的行为特性。通过调用 `shutdown()` `awaitTermination()`,可以确保所有任务完成后再关闭线程池,避免资源泄漏数据不一致的问题。此外,`CountDownLatch` `CompletionService` 提供了更灵活的等待机制,适用于不同的场景。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值