1. Java5之后,并发线程这块发生了根本的变化,最重要的莫过于新的启动、调度、管理线程的一大堆API了。在Java5以后,通过
    Executor来启动线程比用Threadstart()更好。在新特征中,可以很容易控制线程的启动、执行和关闭过程,还可以很容易使用线程池的特
    性。
  2.  
  3. 一、创建任务
  4.  
  5. 任务就是一个实现了Runnable接口的类。
  6.  
  7. 创建的时候实run方法即可。
  8.  
  9. 二、执行任务
  10.  
  11. 通过java.util.concurrent.ExecutorService接口对象来执行任务,该接口对象通过工具类java.util.concurrent.Executors的静态方法来创建。
  12.  
  13. Executors此包中所定义的 ExecutorExecutorServiceScheduledExecutorServiceThreadFactory Callable 类的工厂和实用方法。
  14.  
  15. ExecutorService提供了管理终止的方法,以及可为跟踪一个或多个异步任务执行状况而生成 Future 的方法。 可以关闭
    ExecutorService,这将导致其停止接受新任务。关闭后,执行程序将最后终止,这时没有任务在执行,也没有任务在等待执行,并且无法提交新任
    务。
  16.  
  17. executorService.execute(new TestRunnable());
  18.  
  19. 1、创建ExecutorService
  20.  
  21. 通过工具类java.util.concurrent.Executors的静态方法来创建。
  22.  
  23. Executors此包中所定义的 ExecutorExecutorServiceScheduledExecutorServiceThreadFactory Callable 类的工厂和实用方法。
  24.  
  25. 比如,创建一个ExecutorService的实例,ExecutorService实际上是一个线程池的管理工具:
  26.  
  27. ExecutorService executorService = Executors.newCachedThreadPool();
  28.  
  29. ExecutorService executorService = Executors.newFixedThreadPool(3);
  30.  
  31. ExecutorService executorService = Executors.newSingleThreadExecutor();
  32.  
  33. 2、将任务添加到线程去执行
  34.  
  35. 当将一个任务添加到线程池中的时候,线程池会为每个任务创建一个线程,该线程会在之后的某个时刻自动执行。
  36.  
  37. 三、关闭执行服务对象
  38.  
  39. executorService.shutdown();
  40.  
  41. 四、综合实例
  42.  
  43. package concurrent;
  44.  
  45. import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
  46.  
  47. /**
    * 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);
  48.  
  49. // ExecutorService executorService = Executors.newSingleThreadExecutor();
  50.  
  51. for (int i = 0; i < 5; i++) {
    executorService.execute(new TestRunnable());
    System.out.println("************* a" + i + " *************");
    }
    executorService.shutdown();
    }
    }
  52.  
  53. 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();
    }
    }
    }
    }
  54.  
  55. 运行结果:
  56.  
  57. ************* 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
    ......
  58.  
  59. 五、获取任务的执行的返回值
  60.  
  61. Java5之后,任务分两类:一类是实现了Runnable接口的类,一类是实现了Callable接口的类。两者都可以被
    ExecutorService执行,但是Runnable任务没有返回值,而Callable任务有返回值。并且Callablecall()方法只
    能通过ExecutorService的(<T> task) 方法来执行,并且返回一个 <T><T>,是表示任
    务等待完成的 Future
  62.  
  63. public interface Callable<V>
    返回结果并且可能抛出异常的任务。实现者定义了一个不带任何参数的叫做 call 的方法。
    Callable 接口类似于,两者都是为那些其实例可能被另一个线程执行的类设计的。但是 Runnable 不会返回结果,并且无法抛出经过检查的异常。
    类包含一些从其他普通形式转换成 Callable 类的实用方法。
  64.  
  65. Callable中的call()方法类似Runnablerun()方法,就是前者有返回值,后者没有。
  66.  
  67. 当将一个Callable的对象传递给ExecutorServicesubmit方法,则该call方法自动在一个线程上执行,并且会返回执行结果Future对象。
  68.  
  69. 同样,将Runnable的对象传递给ExecutorServicesubmit方法,则该run方法自动在一个线程上执行,并且会返回执行结果Future对象,但是在该Future对象上调用get方法,将返回null
  70.  
  71. 遗憾的是,在Java API文档中,这块介绍的很糊涂,估计是翻译人员还没搞清楚的缘故吧。或者说是注释不到位。下面看个例子:
  72.  
  73. import java.util.ArrayList;
    import java.util.List;
    import java.util.concurrent.*;
  74.  
  75. /**
    * 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>>();
  76.  
  77. //创建10个任务并执行
    for (int i = 0; i < 10; i++) {
    //使用ExecutorService执行Callable类型的任务,并将结果保存在future变量中
    Future<String> future = executorService.submit(new TaskWithResult(i));
    //将任务执行结果存储到List中
    resultList.add(future);
    }
  78.  
  79. //遍历任务的结果
    for (Future<String> fs : resultList) {
    try {
    System.out.println(fs.get()); //打印各个线程(任务)执行的结果
    } catch (InterruptedException e) {
    e.printStackTrace();
    } catch (ExecutionException e) {
    e.printStackTrace();
    } finally {
    //启动一次顺序关闭,执行以前提交的任务,但不接受新任务。如果已经关闭,则调用没有其他作用。
    executorService.shutdown();
    }
    }
    }
    }
  80.  
  81. class TaskWithResult implements Callable<String> {
    privateint id;
  82.  
  83. public TaskWithResult(int id) {
    this.id = id;
    }
  84.  
  85. /**
    * 任务的具体过程,一旦任务传给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();
    }
    }
  86.  
  87. 运行结果:
    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
  88.  
  89. Process finished with exit code 0
    //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  90.  
  91. 因为之前一直是用的execute方法,最近有个情况需要用到submit方法,所以研究了下。
  92.  
  93. 三个区别:
  94.  
  95. 1、接收的参数不一样
  96.  
  97. 2submit有返回值,而execute没有
  98.  
  99. 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.
  100.  
  101. 用到返回值的例子,比如说我有很多个做validationtask,我希望所有的task执行完,然后每个task告诉我它的执行结果,是成功还是失败,如果是失败,原因是什么。然后我就可以把所有失败的原因综合起来发给调用者。
  102.  
  103. 个人觉得cancel execution这个用处不大,很少有需要去取消执行的。
  104.  
  105. 而最大的用处应该是第二点。
  106.  
  107. 3submit方便Exception处理
  108.  
  109. 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.
  110.  
  111. 意思就是如果你在你的task里会抛出checked或者unchecked exception,而你又希望外面的调用者能够感知这些exception并做出及时的处理,那么就需要用到submit,通过捕获Future.get抛出的异常。

ExecutorService中submit和execute的区别的更多相关文章

  1. ExecutorService中submit()和execute()的区别

    在使用java.util.concurrent下关于线程池一些类的时候,相信很多人和我一样,总是分不清submit()和execute()的区别,今天从源码方面分析总结一下. 通常,我们通过Execu ...

  2. ExecutorService中submit和execute的区别(转)

    在Java5之后,并发线程这块发生了根本的变化,最重要的莫过于新的启动.调度.管理线程的一大堆API了.在Java5以后,通过Executor来启动线程比用Thread的start()更好.在新特征中 ...

  3. ExecutorService中submit和execute的区别<转>

    在Java5之后,并发线程这块发生了根本的变化,最重要的莫过于新的启动.调度.管理线程的一大堆API了.在Java5以后,通过Executor来启动线程比用Thread的start()更好.在新特征中 ...

  4. 多线程ExecutorService中submit和execute区别

    submit和execute都是 ExecutorService 的方法,都是添加线程到线程池中. 区别 三个区别: 1.接收的参数不一样 2.submit有返回值,而execute没有 Method ...

  5. 线程池中 submit()和 execute()方法有什么区别?(未完成)

    线程池中 submit()和 execute()方法有什么区别?(未完成)

  6. 【多线程 5】线程池的类型以及submit()和execute()的区别

    就跟题目说的一样,本篇博客,本宝宝主要介绍两个方面的内容,其一:线程池的类型及其应用场景:其二:submit和execute的区别.那么需要再次重申的是,对于概念性的东西,我一般都是从网上挑选截取,再 ...

  7. 线程池的类型以及执行线程submit()和execute()的区别

    就跟题目说的一样,本篇博客,本宝宝主要介绍两个方面的内容,其一:线程池的类型及其应用场景:其二:submit和execute的区别.那么需要再次重申的是,对于概念性的东西,我一般都是从网上挑选截取,再 ...

  8. html中submit和button的区别(总结) [ 转自欣步同学 ]

    html中submit和button的区别(总结) submit是button的一个特例,也是button的一种,它把提交这个动作自动集成了. 如果表单在点击提交按钮后需要用JS进行处理(包括输入验证 ...

  9. 线程池提交任务时submit()和execute()的区别

    因为之前一直是用的execute方法,最近有个情况需要用到submit方法,所以研究了下. 他们的区别: 1.execut()可以添加一个Runable任务,submit()不仅可以添加Runable ...

随机推荐

  1. php文本操作方法集合比较

    fgets和fputs.fread和fwrite.fscanf和fprintf 字符串读写函数fgets和fputs 一.读字符串函数fgets函数的功能是从指定的文件中读一个字符串到字符数组中,函数 ...

  2. WPF中ListBox的项ListBoxItem被选中的时候Background变化

    使用WPF 中ListBox,点击ListBoxItem的时候,自定义它的背景色,曾经在网上找了一些方法, 不是很理想,后来在StackOverflow上找到了,贴出代码和效果图: 效果图:

  3. nginx+keepalived双主高可用负载均衡

    实验环境及软件版本:CentOS版本: 6.6(2.6.32.-504.el6.x86_64)nginx版本: nginx-1.6.3keepalived版本:keepalived-1.2.7 主LB ...

  4. mac os去除去除.DS_Store文件--使用python和go(原创)

    .DS_Store (英文全称 Desktop Services Store)是一种由苹果公司的Mac OS X操作系统所创造的隐藏文件,目的在于存贮文件夹的自定义属性,例如文件们的图标位置或者是背景 ...

  5. Setting composer minimum stability for your application

    Do you have a confusion of how do you determine the stability when using composer dependency manager ...

  6. phpstorm8 设置及license key

    phpstorm8 license key Learn Programming ===== LICENSE BEGIN ===== 63758-12042010 00000Ryqh0NCC73lpRm ...

  7. [MySql] 设置了UTF8,中文存数据库中仍然出现问号

    运行命令:SHOW VARIABLES LIKE 'character_set_%'; 结果 'character_set_client', 'utf8' 'character_set_connect ...

  8. iOS 进阶 第三天(0326)

    0326 Xib:描述软件界面,轻量级,一般用来描述局部界面 Xib的获取,如下图所示: xib在我们开发的时候叫xib,但如果运行在我们手机里会生成nib.所以xib和nib两种方法来访问xib

  9. Spiral Matrix II

    Spiral Matrix II Given an integer n, generate a square matrix filled with elements from 1 to n2 in s ...

  10. Ado.Net实现简易(省、市、县)三级联动查询,还附加Access数据

    小弟在博客园驻园不久,初来咋到:将最近写的小程序附上,希望各位大牛们吐槽:激发对程序员围观的童鞋们,赶紧加入IT行业,如果你在上海那简称就是SHIT,哈哈题外话,以下开始切入正题: 坐公交车是旁边偶遇 ...