CompletionService
2015-09-20 16:04
351 查看
如果你向Executor提交了一个批处理任务,并且希望在它们完成后获得结果。为此你可以保存与每个任务相关联的Future,然后不断地调用timeout为零的get,来检验Future是否完成。这样做固然可以,但却相当乏味。幸运的是,还有一个更好的方法:完成服务(Completion service)。
CompletionService整合了Executor和BlockingQueue的功能。你可以将Callable任务提交给它去执行,然后使用类似于队列中的take和poll方法,在结果完整可用时获得这个结果,像一个打包的Future。ExecutorCompletionService是实现CompletionService接口的一个类,并将计算任务委托给一个Executor。
ExecutorCompletionService的实现相当直观。它在构造函数中创建一个BlockingQueue,用它去保持完成的结果。计算完成时会调用FutureTask中的done方法。当提交一个任务后,首先把这个任务包装为一个QueueingFuture,它是FutureTask的一个子类,然后覆写done方法,将结果置入BlockingQueue中,take和poll方法委托给了BlockingQueue,poll()是非阻塞的,若目前无结果,返回一个null,take()是阻塞的,若当前无结果,则线程阻塞,直到产生一个结果。
CompletionService整合了Executor和BlockingQueue的功能。你可以将Callable任务提交给它去执行,然后使用类似于队列中的take和poll方法,在结果完整可用时获得这个结果,像一个打包的Future。ExecutorCompletionService是实现CompletionService接口的一个类,并将计算任务委托给一个Executor。
ExecutorCompletionService的实现相当直观。它在构造函数中创建一个BlockingQueue,用它去保持完成的结果。计算完成时会调用FutureTask中的done方法。当提交一个任务后,首先把这个任务包装为一个QueueingFuture,它是FutureTask的一个子类,然后覆写done方法,将结果置入BlockingQueue中,take和poll方法委托给了BlockingQueue,poll()是非阻塞的,若目前无结果,返回一个null,take()是阻塞的,若当前无结果,则线程阻塞,直到产生一个结果。
public class Test { public static void main(String[] args) throws InterruptedException, ExecutionException { ExecutorService executor = Executors.newCachedThreadPool(); //可以是多个CompletionService对象共用一个ExecutorService对象 CompletionService<Integer> comp = new ExecutorCompletionService<Integer>(executor); for (int i = 0; i < 5; i++) { comp.submit(new Task()); } executor.shutdown(); int count = 0, index = 1; while (count < 5) { //取出来的是Future对象而不是Integer对象 Future<Integer> f = comp.poll(); if (f == null) { System.out.println(index + " 没发现有完成的任务"); } else { System.out.println(index + "产生了一个随机数: " + f.get()); count++; } index++; TimeUnit.MILLISECONDS.sleep(500); } } } class Task implements Callable<Integer> { @Override public Integer call() throws Exception { Random rand = new Random(); TimeUnit.SECONDS.sleep(rand.nextInt(7)); return rand.nextInt(); } }
相关文章推荐
- Android 动画之三 Property Animation—— 属性(Property)动画 【Animator提供基类】
- Codeforces 475C Kamal-ol-molk's Painting 模拟
- 剑指offer--整数中1出现的次数(从1到n整数中1出现的次数)
- 隐私泄露杀手锏 —— Flash 权限反射
- win32 线程知识点梳理四
- 指向常量的指针与常量指针的异同
- 关于Cypress公司USB芯片FX2的固件开发
- Can not perform this action after onSaveInstanceState
- 软工第一次文档总结
- MapReduce案例学习(3) 求每个部门最早进入公司的员工姓名
- 关于判断字符串长度的方法
- 关于java内部类访问类的静态成员变量
- HDU-5461 Largest Point
- 在win7系统下使用TortoiseGit(乌龟git)简单操作Git@OSC
- 浅谈搜索引擎优化方法
- js事件知识整理
- 质量的定位没做好,怎么做都是错
- iou-web 无法启动设备的问题
- 1.3信息的表示和处理(学习过程)
- 欧拉工程第68题:Magic 5-gon ring