CompletionService是什么?

它是JUC包中的一个接口类,默认实现类只有一个ExecutorCompletionService。

CompletionService干什么的?

它将异步任务的生成和执行结果的处理进行了解耦,用来执行Callable的任务(实际也是通过Executor线程池执行的,只是它又加了一层封装),我们只需要调用它的take(阻塞)/poll(非阻塞)方法便可以获取到执行完的任务结果,最先获取到的必定是先执行完的异步任务结果。

主要应用场景:同时执行多个Callable任务,并且需对任务的返回结果进行处理。若想优先处理先执行完的任务结果,使用它尤其方便

ExecutorCompletionService 源码解析

有三个成员变量,关键的是下面标注的变量1、变量2:

1 public class ExecutorCompletionService<V> implements CompletionService<V> {
2 private final Executor executor; // 变量1: 线程池
3 private final AbstractExecutorService aes;
4 private final BlockingQueue<Future<V>> completionQueue; // 变量2: 阻塞队列

两个构造器,如下,用于初始化上面的三个成员变量,可以看到Executor线程池是必传的:

1     public ExecutorCompletionService(Executor executor) {
2 if (executor == null)
3 throw new NullPointerException();
4 this.executor = executor;
5 this.aes = (executor instanceof AbstractExecutorService) ?
6 (AbstractExecutorService) executor : null;
7 this.completionQueue = new LinkedBlockingQueue<Future<V>>();
8 }
1     public ExecutorCompletionService(Executor executor,
2 BlockingQueue<Future<V>> completionQueue) {
3 if (executor == null || completionQueue == null)
4 throw new NullPointerException();
5 this.executor = executor;
6 this.aes = (executor instanceof AbstractExecutorService) ?
7 (AbstractExecutorService) executor : null;
8 this.completionQueue = completionQueue;
9 }

核心方法submit,把task封装成一个QueueingFuture,作为执行任务交给线程池执行:

1     public Future<V> submit(Callable<V> task) {
2 if (task == null) throw new NullPointerException();
3 RunnableFuture<V> f = newTaskFor(task);
4 executor.execute(new QueueingFuture(f));
5 return f;
6 }

下面再来看一下QueueingFuture对象,也是一个核心对象,如下所示。QueueingFuture是ExecutorCompletionService的私有内部类,它重写了FutureTask的done()方法。当任务执行完成set值的时候,会调用done方法,在done方法中将task存入阻塞队列。先执行完的任务就会先放入阻塞队列,所以我们从队列中取结果的时候,必定是先取到先执行完的任务。

1     private class QueueingFuture extends FutureTask<Void> {
2 QueueingFuture(RunnableFuture<V> task) {
3 super(task, null);
4 this.task = task;
5 }
6 protected void done() { completionQueue.add(task); }
7 private final Future<V> task;
8 }

总结一下,ExecutorCompletionService是通过QueueingFuture的done方法和阻塞队列实现的按照异步任务返回顺序来返回结果

ExecutorCompletionService和ExecutorService的使用demo示例

Callable类:

 1 class CsCallable implements Callable<String> {
2 private String name;
3 private long milli;
4
5 public CsCallable(String name, long milli) {
6 this.name = name;
7 this.milli = milli;
8 }
9
10 @Override
11 public String call() throws Exception {
12 System.out.println("name:" + name);
13 Thread.sleep(milli);
14 return name + " after " + milli + "ms call back.";
15 }
16 }

ExecutorCompletionService的demo:

 1 public class CompletionServiceDemo {
2 public static void main(String[] args) throws Exception {
3 CompletionService completionService = new ExecutorCompletionService(Executors.newFixedThreadPool(4));
4 completionService.submit(new CsCallable("xxx", 5000));
5 completionService.submit(new CsCallable("www", 2000));
6 completionService.submit(new CsCallable("zzz", 14000));
7 completionService.submit(new CsCallable("yyy", 9000));
8 for (int i = 0; i < 4; i++) {
9 System.out.println(completionService.take().get());
10 }
11 System.out.println("----- main over -----");
12 }
13 }

执行结果如下,可以看到早完成的任务结果能先获取到

ExecutorService的demo:

 1 public class ExecutorServiceDemo {
2 public static void main(String[] args) throws Exception {
3 ExecutorService executorService = Executors.newFixedThreadPool(4);
4 List<Future<String>> list = new ArrayList<>();
5 list.add(executorService.submit(new CsCallable("xxx", 5000)));
6 list.add(executorService.submit(new CsCallable("www", 2000)));
7 list.add(executorService.submit(new CsCallable("zzz", 14000)));
8 list.add(executorService.submit(new CsCallable("yyy", 9000)));
9 for (Future<String> future : list) {
10 System.out.println(future.get());
11 }
12 System.out.println("----- main over -----");
13 }
14 }

执行结果如下,只能按照指定的顺序处理返回结果,无法先处理早完成的任务

Java并发之CompletionService详解的更多相关文章

  1. Java并发之AQS详解

    一.概述 谈到并发,不得不谈ReentrantLock:而谈到ReentrantLock,不得不谈AbstractQueuedSynchronizer(AQS)! 类如其名,抽象的队列式的同步器,AQ ...

  2. Java并发之AQS详解(转)

    一.概述 谈到并发,不得不谈ReentrantLock:而谈到ReentrantLock,不得不谈AbstractQueuedSynchronized(AQS)! 类如其名,抽象的队列式的同步器,AQ ...

  3. Java并发之ReentrantLock详解

    一.入题 ReentrantLock是Java并发包中互斥锁,它有公平锁和非公平锁两种实现方式,以lock()为例,其使用方式为: ReentrantLock takeLock = new Reent ...

  4. java并发之CAS详解

    前言 在高并发的应用当中,最关键的问题就是对共享变量的安全访问,通常我们都是通过加锁的方式,比如说synchronized.Lock来保证原子性,或者在某些应用当中,用voliate来保证变量的可见性 ...

  5. java并发之synchronized详解

    前言 多个线程访问同一个类的synchronized方法时, 都是串行执行的 ! 就算有多个cpu也不例外 ! synchronized方法使用了类java的内置锁, 即锁住的是方法所属对象本身. 同 ...

  6. Android开发之InstanceState详解

    Android开发之InstanceState详解   本文介绍Android中关于Activity的两个神秘方法:onSaveInstanceState() 和 onRestoreInstanceS ...

  7. Android开发之InstanceState详解(转)---利用其保存Activity状态

    Android开发之InstanceState详解   本文介绍Android中关于Activity的两个神秘方法:onSaveInstanceState() 和 onRestoreInstanceS ...

  8. Android开发之MdiaPlayer详解

    Android开发之MdiaPlayer详解 MediaPlayer类可用于控制音频/视频文件或流的播放,我曾在<Android开发之基于Service的音乐播放器>一文中介绍过它的使用. ...

  9. Java 8 Stream API详解--转

    原文地址:http://blog.csdn.net/chszs/article/details/47038607 Java 8 Stream API详解 一.Stream API介绍 Java8引入了 ...

随机推荐

  1. P2617 Dynamic Rankings (动态开点权值线段树 + 树状数组)

    题意:带修求区间k小 题解:回忆在使用主席树求区间k小时 利用前缀和的思想 既然是前缀和 那么我们可以使用更擅长维护前缀和的树状数组 但是这里每一颗权值线段树就不是带版本的 而是维护数组里i号点的权值 ...

  2. Color Changing Sofa Gym - 101962B、Renan and Cirque du Soleil Gym - 101962C、Hat-Xor Gym - 101962E 、Rei do Cangaço Gym - 101962K 、Sorting Machine Gym - 101962M

    Color Changing Sofa Gym - 101962B 题意:给你一个由字母构成的字符串a,再给你一个由0.1构成的字符串b.你需要在a字符串中找到一个可以放下b的位置,要保证b字符串中0 ...

  3. 【noi 2.6_2000】&【poj 2127】 最长公共子上升序列 (DP+打印路径)

    由于noi OJ上没有Special Judge,所以我是没有在这上面AC的.但是在POJ上A了. 题意如标题. 解法:f[i][j]表示a串前i个和b串前j个且包含b[j]的最长公共上升子序列长度 ...

  4. Codeforces Round #652 (Div. 2)D. TediousLee 推导

    题意: Rooted Dead Bush (RDB) of level 1是只有一个点,如下图 当(RDB) of level i变成(RDB) of level i+1的时候,每一个顶点要进行下面的 ...

  5. 牛客小白月赛17 G 区间求和

    传送门 题意: 题解: 原本想着使用暴力方法: 1 #include<stdio.h> 2 #include<string.h> 3 #include<iostream& ...

  6. poj1180 Batch Scheduling

    Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 3590   Accepted: 1654 Description There ...

  7. Codeforces Round #547 (Div. 3) D. Colored Boots (贪心,模拟)

    题意:有两个字符串,两个字符串中的相同字符可以相互匹配,\(?\)可以和任意字符匹配,输出最大匹配的字符数量和它们分别两个字符串中的位置. 题解:很容易贪心,我们先遍历第一个字符串,然后在第二个字符串 ...

  8. 5.Fanout交换机之新订单通知商户场景

    标题 : 5.Fanout交换机之新订单通知商户场景 目录 : RabbitMQ 序号 : 5 const string newOrderQueueName = "neworder-queu ...

  9. python3基本数据类型补充

    列表 list 有序,可嵌套,可重复,元素可修改 方括号 占用空间小但时间消耗比较大 mylist=["kimi",1,1,1,["amy",18]] V=my ...

  10. sdut3562-求字典序最小的最短路 按顶点排序后spfa的反例

    首先我们可以这么搞...倒序建图,算出源点s附近的点距离终点的距离,然后判断一下,终点是否能跑到源点 能跑到的话呢,我们就判断s周围的点是否在最短路上,然后我们选编号最小的点就好了 代码 #inclu ...