ExecutorService中submit和execute的区别
来源:互联网 发布:数据库存储多个坐标 编辑:程序博客网 时间:2024/05/01 00:44
一、创建任务
二、执行任务
1、创建ExecutorService
2、将任务添加到线程去执行
三、关闭执行服务对象
四、综合实例
package concurrent; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; /** * Created by IntelliJ IDEA. * * @author leizhimin 2008-11-25 14:28:59 */publicclass TestCachedThreadPool { publicstaticvoid main(String[] args) { // ExecutorService executorService = Executors.newCachedThreadPool(); ExecutorService executorService = Executors.newFixedThreadPool(5);// ExecutorService executorService = Executors.newSingleThreadExecutor(); for (int i = 0; i < 5; i++) { executorService.execute(new TestRunnable()); System.out.println("************* a" + i + " *************"); } executorService.shutdown(); } } class TestRunnable implements Runnable { publicvoid run() { System.out.println(Thread.currentThread().getName() + "线程被调用了。"); while (true) { try { Thread.sleep(5000); System.out.println(Thread.currentThread().getName()); } catch (InterruptedException e) { e.printStackTrace(); } } } }
运行结果:************* a0 ************* ************* a1 ************* pool-1-thread-2线程被调用了。 ************* a2 ************* pool-1-thread-3线程被调用了。 pool-1-thread-1线程被调用了。 ************* a3 ************* ************* a4 ************* pool-1-thread-4线程被调用了。 pool-1-thread-5线程被调用了。 pool-1-thread-2 pool-1-thread-1 pool-1-thread-3 pool-1-thread-5 pool-1-thread-4 pool-1-thread-2 pool-1-thread-1 pool-1-thread-3 pool-1-thread-5 pool-1-thread-4 ......
五、获取任务的执行的返回值
import java.util.ArrayList; import java.util.List; import java.util.concurrent.*; /** * Callable接口测试 * * @author leizhimin 2008-11-26 9:20:13 */publicclass CallableDemo { publicstaticvoid main(String[] args) { ExecutorService executorService = Executors.newCachedThreadPool(); List<Future<String>> resultList = new ArrayList<Future<String>>(); //创建10个任务并执行 for (int i = 0; i < 10; i++) { //使用ExecutorService执行Callable类型的任务,并将结果保存在future变量中 Future<String> future = executorService.submit(new TaskWithResult(i)); //将任务执行结果存储到List中 resultList.add(future); } //遍历任务的结果 for (Future<String> fs : resultList) { try { System.out.println(fs.get()); //打印各个线程(任务)执行的结果 } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } finally { //启动一次顺序关闭,执行以前提交的任务,但不接受新任务。如果已经关闭,则调用没有其他作用。 executorService.shutdown(); } } } }
class TaskWithResult implements Callable<String> { privateint id; public TaskWithResult(int id) { this.id = id; } /** * 任务的具体过程,一旦任务传给ExecutorService的submit方法,则该方法自动在一个线程上执行。 * * @return * @throws Exception */ public String call() throws Exception { System.out.println("call()方法被自动调用,干活!!! " + Thread.currentThread().getName()); //一个模拟耗时的操作 for (int i = 999999; i > 0; i--) ; return"call()方法被自动调用,任务的结果是:" + id + " " + Thread.currentThread().getName(); } }
运行结果:call()方法被自动调用,干活!!! pool-1-thread-1 call()方法被自动调用,干活!!! pool-1-thread-3 call()方法被自动调用,干活!!! pool-1-thread-4 call()方法被自动调用,干活!!! pool-1-thread-6 call()方法被自动调用,干活!!! pool-1-thread-2 call()方法被自动调用,干活!!! pool-1-thread-5 call()方法被自动调用,任务的结果是:0 pool-1-thread-1 call()方法被自动调用,任务的结果是:1 pool-1-thread-2 call()方法被自动调用,干活!!! pool-1-thread-2 call()方法被自动调用,干活!!! pool-1-thread-6 call()方法被自动调用,干活!!! pool-1-thread-4 call()方法被自动调用,任务的结果是:2 pool-1-thread-3 call()方法被自动调用,干活!!! pool-1-thread-3 call()方法被自动调用,任务的结果是:3 pool-1-thread-4 call()方法被自动调用,任务的结果是:4 pool-1-thread-5 call()方法被自动调用,任务的结果是:5 pool-1-thread-6 call()方法被自动调用,任务的结果是:6 pool-1-thread-2 call()方法被自动调用,任务的结果是:7 pool-1-thread-6 call()方法被自动调用,任务的结果是:8 pool-1-thread-4 call()方法被自动调用,任务的结果是:9 pool-1-thread-3 Process finished with exit code 0
因为之前一直是用的execute方法,最近有个情况需要用到submit方法,所以研究了下。
三个区别:
1、接收的参数不一样
2、submit有返回值,而execute没有
Method submit extends base method Executor.execute by creating and returning a Future that can be used to cancel execution and/or wait for completion.
用到返回值的例子,比如说我有很多个做validation的task,我希望所有的task执行完,然后每个task告诉我它的执行结果,是成功还是失败,如果是失败,原因是什么。然后我就可以把所有失败的原因综合起来发给调用者。
个人觉得cancel execution这个用处不大,很少有需要去取消执行的。
而最大的用处应该是第二点。
3、submit方便Exception处理
There is a difference when looking at exception handling. If your tasks throws an exception and if it was submitted with execute
this exception will go to the uncaught exception handler (when you don't have provided one explicitly, the default one will just print the stack trace to System.err
). If you submitted the task with submit
any thrown exception, checked or not, is then part of the task's return status. For a task that was submitted with submit
and that terminates with an exception, the Future.get
will rethrow this exception, wrapped in an ExecutionException
.
意思就是如果你在你的task里会抛出checked或者unchecked exception,而你又希望外面的调用者能够感知这些exception并做出及时的处理,那么就需要用到submit,通过捕获Future.get抛出的异常。
比如说,我有很多更新各种数据的task,我希望如果其中一个task失败,其它的task就不需要执行了。那我就需要catch Future.get抛出的异常,然后终止其它task的执行,
// 创建10个任务并执行 for (int i = 0; i < 10; i++) { // 使用ExecutorService执行Callable类型的任务,并将结果保存在future变量中 Future<String> future = executorService.submit(new TaskWithResult(i)); // 将任务执行结果存储到List中 resultList.add(future); } executorService.shutdown(); // 遍历任务的结果 for (Future<String> fs : resultList) { try { System.out.println(fs.get()); // 打印各个线程(任务)执行的结果 } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { executorService.shutdownNow(); e.printStackTrace(); return; } }
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit和execute的区别
- ExecutorService中submit()和execute()的区别
- ExecutorService中submit和execute的区别
- Java-类加载器笔记
- 黑马程序员-自己总结的java学习笔记(3)java字符串对象。
- mysql启动提示mysql.host 不存在,启动失败的解决方法
- 简单介绍VC++6.0下如何使用压缩库ZLIB
- jqGrid 行编辑 select 3级联动 的一种实现方法
- ExecutorService中submit和execute的区别
- 获取android 手机屏幕的大小以及动态设置背景图片的两种方法;
- 如何在iOS中使用ZXing库
- 卫星影像的分辨率与成图比例尺
- unite2014参会后的个人总结
- Tomcat服务器常用配置(一)
- python 中的 __new__ 和 __init__
- [centos5.4 x86 64位]使用第三方软件仓库|使用RPMForge软件库
- Oracle 多个连接池