并发编程(六)-AbstractExecutorService源码分析

一、AbstractExecutorService简介


AbstractExecutorService是一个抽象类,实现了ExecutorService接口,提供了线程池的基本实现。它是Java Executor框架的核心类,提供了线程池的基本操作,如提交任务、管理线程池、执行任务等。

自定义执行器服务可用扩展AbreactExecutorService 并覆盖其方法以提供自己的实现。这使开发人员可以创建适合其特定需求的执行器服务。

二、AbstractExecutorService如何使用


AbstractExecutorService是一个抽象类,不能直接实例化,需要通过继承它的子类ThreadPoolExecutorScheduledThreadPoolExecutor来使用。使用AbstractExecutorService创建线程池的步骤如下:

2.1 没有返回值

// 1、创建ThreadPoolExecutor  创建一个固定大小的线程池,核心线程数为1,最大线程数为5,任务队列大小为10
ExecutorService executorService = new ThreadPoolExecutor(1, 5, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue(10));
// 2、提交任务
executorService.submit(new Runnable() {
    @Override
    public void run() {
        //执行任务
    }
});
// 3、关闭线程池
executorService.shutdown();

2.2 有返回值

// 1、创建ThreadPoolExecutor  创建一个固定大小的线程池,核心线程数为1,最大线程数为5,任务队列大小为10
ExecutorService executorService = new ThreadPoolExecutor(1, 5, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue(10));
// 2、提交一个Callable任务
Future future = executorService.submit(new Callable() {
    @Override
    public String call() throws Exception {
        //执行任务,返回结果
        return "hello ExecutorService";
    }
});
// 3、获取值
String result = future.get();
// 4、关闭线程池
executorService.shutdown();

总之,使用AbstractExecutorService创建线程池非常简单,只需要创建线程池对象、提交任务、关闭线程池即可。同时,AbstractExecutorService也提供了一些高级功能,如异步任务、定时任务等,可以根据具体需求选择使用。

三、如何自定义AbstractExecutorService


要自定义一个延迟AbstractExecutorService,以下步骤:

  1. 继承AbstractExecutorService类

  1. 实现 submit 方法 覆盖 submit 方法,将任务添加到队列中,其中在任务执行前应用延迟。

  1. 实现 execute 方法 覆盖 execute 方法,调用 submit 方法并返回 Future 对象,等待任务完成。

  1. 实现shutdown 方法 覆盖 shutdown 方法,停止所有任务并等待它们完成。

  1. 实现 DelayedTask 类 创建一个 DelayedTask 类,用于封装任务和延迟时间。

  1. 实现 RunnableTask 和 CallableTask 类 创建 RunnableTask 和 CallableTask 类,用于封装任务。

public class CustomDelayedExecutorService extends AbstractExecutorService {
    
    private final DelayQueue> queue = new DelayQueue<>();
    
    @Override
    public  Future submit(Callable task) {
        return schedule(new CallableTask<>(task));
    }
    
    @Override
    public Future submit(Runnable task) {
        return schedule(new RunnableTask(task));
    }
    
    private  Future schedule(DelayedTask task) {
        queue.offer(task);
        return task;
    }
    
    @Override
    public void execute(Runnable command) {
        submit(command);
    }
    
    @Override
    public void shutdown() {
        for (DelayedTask task : queue) {
            task.cancel(false);
        }
    }
    
    @Override
    public List shutdownNow() {
        List tasks = new ArrayList<>();
        for (DelayedTask task : queue) {
            if (task.cancel()) {
                tasks.add(task.getTask());
            }
        }
        return tasks;
    }
    
    @Override
    public boolean isShutdown() {
        return false;
    }
    
    @Override
    public boolean isTerminated() {
        return false;
    }
    
    @Override
    public boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException {
        return false;
    }
​
}
class DelayedTask implements RunnableFuture, Delayed {
    private final long delay;
    private final long expire;
    private final RunnableFuture task;
    public DelayedTask(RunnableFuture task, long delay) {
        this.task = task;
        this.delay = delay;
        this.expire = System.nanoTime() + TimeUnit.NANOSECONDS.convert(delay, TimeUnit.MILLISECONDS);
    }
   
    @Override
    public long getDelay(TimeUnit unit) {
        return unit.convert(expire - System.nanoTime(), TimeUnit.NANOSECONDS);
    }
   
    @Override
    public int compareTo(Delayed o) {
        return Long.compare(expire, ((DelayedTask) o).expire);
    }
   
    @Override
    public void run() {
        if (!isCancelled()) {
            task.run();
        }
    }
   
    @Override
    public boolean cancel(boolean mayInterruptIfRunning) {
        return task.cancel(mayInterruptIfRunning);
    }
   
    @Override
    public boolean isCancelled() {
        return task.isCancelled();
    }
   
    @Override
    public boolean isDone() {
        return task.isDone();
    }
   
    @Override
    public T get() throws InterruptedException, ExecutionException {
        return task.get();
    }
   
    @Override
    public T get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
        return task.get(timeout, unit);
    }
   
    public RunnableFuture getTask() {
        return task;
    }
}
​
class RunnableTask implements RunnableFuture {
    private final Runnable task;
    public RunnableTask(Runnable task) {
        this.task = task;
    }
    @Override
    public void run() {
        task.run();
    }
    @Override
    public boolean cancel(boolean mayInterruptIfRunning) {
        return false;
    }
    @Override
    public boolean isCancelled() {
        return false;
    }
    @Override
    public boolean isDone() {
        return true;
    }
    @Override
    public Void get() throws InterruptedException, ExecutionException {
        return null;
    }
    @Override
    public Void get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
        return null;
    }
}
class CallableTask extends FutureTask {
    public CallableTask(Callable callable) {
        super(callable);
    }
}

四、AbstractExecutorService源码分析


4.1 AbstractExecutorService dragrams

并发编程(六)-AbstractExecutorService源码分析_第1张图片

4.2 方法分析

并发编程(六)-AbstractExecutorService源码分析_第2张图片

4.2.1 newTaskFor(Runnable runnable, T value) 方法

AbstractExecutorService中的newTaskFor方法,用于创建一个RunnableFuture对象,其中RunnableFuture是Future和Runnable接口的结合体,可以用来表示异步执行的结果。该方法的参数包括一个Runnable对象和一个泛型T的值。通过传入的Runnable对象和T类型的值,创建了一个FutureTask对象,并将其返回。

FutureTask是RunnableFuture的一个具体实现,它表示一个可取消的异步计算,可以通过调用get方法获取计算结果。在创建FutureTask对象时,需要传入一个Callable对象或Runnable对象,用于执行异步计算。而在这里,我们传入了一个Runnable对象和一个T类型的值,表示该Runnable对象的执行结果为T类型的值。

protected  RunnableFuture newTaskFor(Runnable runnable, T value) {
      // 将runnable包装成FutureTask类型
      return new FutureTask(runnable, value);
 }

4.2.2 submit(Runnable task)方法

并发编程(六)-AbstractExecutorService源码分析_第3张图片

AbstractExecutorService中的submit方法是ExecutorService接口中的一个方法,用于提交一个Runnable任务,并返回一个表示该任务异步执行结果的Future对象。

  1. 如果传入的task为null,则抛出NullPointerException异常;

  1. 调用newTaskFor(Runnable runnable, T value)方法创建一个RunnableFuture对象ftask,其中RunnableFuture是Future和Runnable接口的结合体,用于表示异步执行的结果。在该方法中,我们将泛型T的值设置为null,表示该Runnable任务没有返回值;

  1. 通过调用execute(Runnable command)方法提交任务进行执行,该方法是一个抽象方法,需要由子类实现。在该方法中,我们将创建的ftask对象作为参数传入,表示需要执行的任务为ftask对象。

  1. 最后,将ftask对象返回,作为Future对象,用于获取任务执行结果或取消任务。

public Future submit(Runnable task) {
      // 如果任务为null,则抛出nullPointerException
      if (task == null) throw new NullPointerException();
      // 将task包装成RunnableFuture
      RunnableFuture ftask = newTaskFor(task, null);
      // 通过调用execute(Runnable command)方法提交任务进行执行
      execute(ftask);
      // 返回RunnableFuture
      return ftask;
 }

4.2.3 doInvokeAny(Collection> tasks,boolean timed, long nanos)方法

并发编程(六)-AbstractExecutorService源码分析_第4张图片

AbstractExecutorService中doInvokeAny方法用于执行一个任务集合中的所有任务并返回第一个成功执行的任务的结果,如果所有任务都执行失败则抛出ExecutionException异常。其中,方法参数tasks为任务集合,timed和nanos用于指定超时时间,如果超时则抛出TimeoutException异常。

  1. 如果任务集合是否为空,则抛出NullPointerException异常

  1. 如果任务集合的大小为0,则抛出IllegalArgumentException异常

  1. futures用于存储所有任务的Future对象, ecs对象用于异步执行任务集合中的任务。

  1. 从迭代器中获取一个任务xecutorCompletionService,并将该任务的执行结果的Future添加到futures列表中

  1. 死循环不断从ExecutorCompletionService中获取已完成的任务的结果Future,直到有一个任务完成为止,具体细流程如下:

  • 调用ExecutorCompletionService的poll()方法获取一个已完成的任务的结果Future,如果poll()返回null,表示当前没有已完成的任务。

  • 当前没有已完成的任务

  • 还有未执行的任务(ntasks > 0),则从任务集合中获取下一个任务,并将其提交给ExecutorCompletionService执行。提交成功后,活跃线程数active加一;

  • 没有执行的任务,并且活跃线程数为0(即所有任务都已执行完毕),则退出循环;

  • 指定超时时间(timed为true),则调用poll(nanos, TimeUnit.NANOSECONDS)方法获取已完成的任务的结果Future,并等待最多nanos纳秒的时间。如果超时则抛出TimeoutException;

  • 没有指定超时时间,则调用take()方法阻塞等待已完成的任务的结果Future。

  • 如果获取到了已完成的任务的结果Future,则将活跃线程数active减一,并尝试获取该任务的执行结果。

  • 如果获取结果时出现异常,则将异常保存在ExecutionException中,继续下一轮循环;

  • 如果成功获取到了一个任务的执行结果,则直接返回该结果。

  1. 如果所有任务都已经完成,但是没有找到任何一个成功的任务,则抛出ExecutionException异常。

  1. 最后,取消所有尚未完成的任务,以便节省资源并提高效率

private T doInvokeAny(Collection> tasks,
                            boolean timed, long nanos)
      throws InterruptedException, ExecutionException, TimeoutException {
     // 如果任务集合是否为空,则抛出NullPointerException异常
      if (tasks == null)
          throw new NullPointerException();
      int ntasks = tasks.size();
      // 如果任务集合的大小为0,则抛出IllegalArgumentException异常
      if (ntasks == 0)
          throw new IllegalArgumentException();
      // futures用于存储所有任务的Future对象   
      ArrayList> futures = new ArrayList>(ntasks);
      ExecutorCompletionService ecs =
          new ExecutorCompletionService(this);
      try {
          ExecutionException ee = null;
          // 计算deadline时间
          final long deadline = timed ? System.nanoTime() + nanos : 0L;
          Iterator> it = tasks.iterator();
          // 从迭代器中获取一个任务xecutorCompletionService,并将该任务的执行结果的Future添加到futures列表中。
          futures.add(ecs.submit(it.next()));
          // 任务数减1
          --ntasks;
          // 退出循环标识(也就是调用ecs.sumbit的任务数)
          int active = 1;
          // 自旋、死循环
          for (;;) {
              // 获取Future对象
              Future f = ecs.poll();
              // 判断future 任务是否已完成
              if (f == null) {
                 // 任务没有完成,继续判断是否还有未执行的任务
                  if (ntasks > 0) {
                      // 任务s数减1
                      --ntasks;
                      // 再从tasks中获取一个任务,提交到ExecutorCompletionService中进行执行。
                      futures.add(ecs.submit(it.next()));
                     // 提交ecs中加1
                      ++active;
                  }
                  else if (active == 0)
                      break;
                  // 有超时时间限制    
                  else if (timed) {
                  //调用ExecutorCompletionService的poll(long timeout, TimeUnit unit)方法等待指定的超时时间。
                      f = ecs.poll(nanos, TimeUnit.NANOSECONDS);
                     // 在等待过程中已经超过了指定的超时时间,因此会抛出 TimeoutException 异常
                      if (f == null)
                          throw new TimeoutException();
                      // 将剩余的超时时间重新计算,并继续执行后续的代码。
                      nanos = deadline - System.nanoTime();
                  }
                  // 如果没有超时时间限制,则调用ExecutorCompletionService的take()方法一直等待,直到有任务执行完成。
                  else
                      f = ecs.take(); 
              }
              // 任务已完成(获取到Future对象)
              if (f != null) {
                  --active;
                  try {
                    // 尝试获取Future对象的执行结果,
                      return f.get();
                  } catch (ExecutionException eex) {
                  // 如果获取失败,则将异常保存在ee变量中,继续等待下一个任务的执行结果
                      ee = eex;
                  } catch (RuntimeException rex) {
                      ee = new ExecutionException(rex);
                  }
              }
          }
          // 如果所有任务都已经完成,但是没有找到任何一个成功的任务,则抛出ExecutionException异常。
          if (ee == null)
              ee = new ExecutionException();
          throw ee;

      } finally {
         // 取消所有尚未完成的任务,以便节省资源并提高效率
          for (int i = 0, size = futures.size(); i < size; i++)
              futures.get(i).cancel(true);
      }
  }

4.2.4 invokeAll(Collection> tasks)方法

并发编程(六)-AbstractExecutorService源码分析_第5张图片

AbstractExecutorService中的invokeAll方法,它的作用是在执行给定的任务集合tasks中的所有任务,并等待所有任务完成后返回一个包含Future对象的列表,Future对象可以用来获取每个任务的执行结果。如果其中某个任务抛出异常,则该异常将传播到调用者。

  • 判断任务是否为空,如果空,则抛出NullPointerException异常;

  • 遍历参数tasks中的每个Callable任务,将每个任务转换成一个RunnableFuture对象,并把任务添加到futures中,然后调用execute方法执行任务。

  • 循环遍历futures中的每个Future对象,如果某个Future对象的任务还没有完成,则调用get()方法等待任务完成,如果任务抛出异常,则忽略异常。

  • 如果所有的任务都成功完成,则将done标记为true,并返回futures;

  • 否则,取消所有未完成的任务,并抛出InterruptedException异常。

public  List> invokeAll(Collection> tasks)
    throws InterruptedException {
    // 判断任务是否为空,如果空,则抛出NullPointerException异常;
    if (tasks == null)
        throw new NullPointerException();
    ArrayList> futures = new ArrayList>(tasks.size());
    boolean done = false;
    try {
        // 遍历参数tasks中的每个Callable任务
        for (Callable t : tasks) {
            // 将任务转换成一个RunnableFuture对象
            RunnableFuture f = newTaskFor(t);
            futures.add(f);
            // 调用execute方法执行任务
            execute(f);
        }
        // 循环遍历futures中的每个Future对象
        for (int i = 0, size = futures.size(); i < size; i++) {
            Future f = futures.get(i);
            // 如果某个Future对象的任务还没有完成
            if (!f.isDone()) {
                try {
                    // 调用get()方法等待任务完成,如果任务抛出异常,则忽略异常。
                    f.get();
                } catch (CancellationException ignore) {
                } catch (ExecutionException ignore) {
                }
            }
        }
        // 如果所有的任务都成功完成,则将done标记为true,并返回futures;
        done = true;
        return futures;
    } finally {
        if (!done)
           // 如果所有的任务没有全部成功完成,取消所有未完成的任务
            for (int i = 0, size = futures.size(); i < size; i++)
                futures.get(i).cancel(true);
    }
}

你可能感兴趣的:(并发编程源码,java)