回顾:

接上篇博客

java线程——三种创建线程的方式,这篇博客主要介绍第三种方式Callable和Future。比较继承Thread类和实现Runnable接口,接口更加灵活,使用更广泛。但这两种方式都没有返回值,要想返回相应的数据,就要使用Callable和Future方式。

基础:

1、Callable

还是从定义开始,Callable接口有返回值,并且可以抛出异常。


  1. /**(有返回值的任务,可能抛出异常)
  2. * A task that returns a result and may throw an exception.
  3. * Implementors define a single method with no arguments called
  4. * {@code call}.
  5. * @see Executor
  6. * @since 1.5
  7. * @author Doug Lea
  8. * @param <V> the result type of method {@code call}
  9. */
  10. @FunctionalInterface
  11. public interface Callable<V> {
  12. V call() throws Exception;
  13. }


2、Future

Future同样也是一个接口,主要方法如下,方法的功能比较容易理解,所以就没有写注释。主要作用:获取任务执行结果,中断任务等。


  1. package java.util.concurrent;
  2. public interface Future<V> {
  3. boolean cancel(boolean mayInterruptIfRunning);
  4. boolean isCancelled();
  5. boolean isDone();
  6. V get() throws InterruptedException, ExecutionException;
  7. V get(long timeout, TimeUnit unit)
  8. throws InterruptedException, ExecutionException, TimeoutException;
  9. }

3、FutureTask


  1. public class FutureTask<V> implements RunnableFuture<V> {
  2. ......
  3. }
  4. public interface RunnableFuture<V> extends Runnable, Future<V> {
  5. ......
  6. }

FutureTask实现了RunnableFuture接口,而RunnableFuture接口继承了Runnable和Future。也就是说,它既可以作为Runnable被线程执行,也可以作为Future得到Callable返回值。

使用:

方法一:Callable+Future


  1. public class CallableAndFuture {
  2. /**
  3. * 实现Callable接口
  4. *
  5. * @author YANG
  6. *
  7. */
  8. public static class MyCallable implements Callable {
  9. private int flag = 0;
  10. public MyCallable(int flag) {
  11. this.flag = flag;
  12. }
  13. // 重写call方法
  14. public String call() throws Exception {
  15. // 情况一:flag=0 返回0
  16. if (this.flag == 0) {
  17. return "flag = 0";
  18. }
  19. // 情况二:flag=1 返回looping 陷入死循环
  20. if (this.flag == 1) {
  21. try {
  22. while (true) {
  23. System.out.println("looping.");
  24. Thread.sleep(2000);
  25. }
  26. // 情况三:出现异常
  27. } catch (InterruptedException e) {
  28. System.out.println("Interrupted");
  29. }
  30. return "false";
  31. } else {
  32. throw new Exception("Bad flag value!");
  33. }
  34. }
  35. }
  36. public static void main(String[] args) {
  37. // 定义3个Callable类型的任务,构造方法中制定flag的值
  38. MyCallable task1 = new MyCallable(0);
  39. MyCallable task2 = new MyCallable(1);
  40. MyCallable task3 = new MyCallable(2);
  41. // 创建一个执行任务的服务
  42. ExecutorService es = Executors.newFixedThreadPool(3);
  43. try {
  44. // 提交并执行任务,任务启动时返回了一个Future对象,
  45. // 如果想得到任务执行的结果或者是异常可对这个Future对象进行操作
  46. Future future1 = null;
  47. future1 = es.submit(task1);
  48. // 获得第一个任务的结果,如果调用get方法,当前线程会等待任务执行完毕后才往下执行
  49. System.out.println("task1: " + future1.get());
  50. Future future2 = es.submit(task2);
  51. // 等待5秒后,再停止第二个任务。因为第二个任务进行的是无限循环
  52. Thread.sleep(5000);
  53. System.out.println("task2 cancel: " + future2.cancel(true));
  54. // 测试抛出异常
  55. Future future3 = es.submit(task3);
  56. System.out.println("task3: " + future3.get());
  57. } catch (Exception e) {
  58. System.out.println(e.toString());
  59. }
  60. // 停止任务执行服务
  61. es.shutdownNow();
  62. }
  63. }

执行结果:

方法二:Callable+FutureTask

分析两种方法不同之处就在于Future和FutureTask,其中一个是接口,一个是类。因此,只有main方法调用部分不同,上面的MyCallable类中的内容保持不变。


  1. public static void main(String[] args) {
  2. MyCallable task1 = new MyCallable(0);
  3. FutureTask ft1 = new FutureTask(task1);
  4. MyCallable task2 = new MyCallable(1);
  5. FutureTask ft2 = new FutureTask(task2);
  6. MyCallable task3 = new MyCallable(2);
  7. FutureTask ft3 = new FutureTask(task3);
  8. try {
  9. //启动task1
  10. new Thread(ft1, "子线程").start();
  11. System.out.println(ft1.get());
  12. //等待5秒后,停止task2
  13. new Thread(ft2, "子线程").start();
  14. Thread.sleep(5000);
  15. System.out.println("task2 cancel:" + ft2.cancel(true));
  16. //启动task3
  17. new Thread(ft3, "子线程").start();
  18. System.out.println("task3:" + ft3.get());
  19. } catch (InterruptedException | ExecutionException e) {
  20. System.out.println(e.toString());
  21. }
  22. }

其执行结果与方法一完全相同,对比这两种方式,第二种比较容易读懂,第一种相对困难些。下篇博客我们介绍Executor、ExecutorService等内容,相信之后理解起来就会很轻松了。

java线程——详解Callable、Future和FutureTask的更多相关文章

  1. Java线程池(Callable+Future模式)

    转: Java线程池(Callable+Future模式) Java线程池(Callable+Future模式) Java通过Executors提供四种线程池 1)newCachedThreadPoo ...

  2. java线程详解

    Java线程:概念与原理 一.操作系统中线程和进程的概念 现在的操作系统是多任务操作系统.多线程是实现多任务的一种方式. 进程是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,一个进程 ...

  3. Java线程详解----借鉴

    Java线程:概念与原理 一.操作系统中线程和进程的概念 现在的操作系统是多任务操作系统.多线程是实现多任务的一种方式. 进程是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,一个进程 ...

  4. Java线程池 / Executor / Callable / Future

    为什么需要线程池?   每次都要new一个thread,开销大,性能差:不能统一管理:功能少(没有定时执行.中断等).   使用线程池的好处是,可重用,可管理.   Executor     4种线程 ...

  5. 【转】Java线程详解

    Java线程:概念与原理 一.操作系统中线程和进程的概念 现在的操作系统是多任务操作系统.多线程是实现多任务的一种方式. 进程是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,一个进程 ...

  6. java线程详解(三)

    java线程间通信 首先看一段代码 class Res { String name; String sex; } class Input implements Runnable { private R ...

  7. 并发编程 || Java线程详解

    通用线程模型 在很多研发当中,实际应用是基于一个理论再进行优化的.所以,在了解JVM规范中的Java线程的生命周期之前,我们可以先了解通用的线程生命周期,这有助于我们后续对JVM线程生命周期的理解. ...

  8. java线程详解(二)

    1,线程安全 先看上一节程序,我们稍微改动一下: //线程安全演示 //火车站有16张票,需要从四个窗口卖出,如果按照上面的多线程实现,程序如下 class Ticket implements Run ...

  9. java线程详解(一)

    1,相关概念简介 (1)进程:是一个正在执行的程序.每一个进程执行都有一个执行的顺序,该顺序就是一个执行路径,或者叫一个控制单元.用于分配空间. (2)线程:就是进程中一个独立的控制单元,线程在控制着 ...

随机推荐

  1. mysql分页小结

    mysql.select('*').from('books') .join('cSessionInfo', 'books.openid', 'cSessionInfo.open_id') .limit ...

  2. BZOJ3645: Maze(FFT多项式快速幂)

    Description 众维拉先后在中土大陆上创造了精灵.人类以及矮人,其中矮人是生性喜好常年居住在地下的洞穴的存在,他们挖掘矿物甚至宝石,甚至用他们的勤劳勇敢智慧在地底下创造出了辉煌宏大的宫殿,错综 ...

  3. nginx 实现跨域

    nginx 添加头部跨域. location / { add_header 'Access-Control-Allow-Origin' '*'; //允许的域 add_header 'Access-C ...

  4. 兼容IE浏览器的canvas画线和圆圈

    1.新建test.html文件,代码如下: <!DOCTYPE html><html><head>    <meta charset="utf-8& ...

  5. LRJ入门经典-0905邮票和信封305

    原题 LRJ入门经典-0905邮票和信封305 难度级别:B: 运行时间限制:1000ms: 运行空间限制:256000KB: 代码长度限制:2000000B 试题描述 假定一张信封最多贴5张邮票,如 ...

  6. 【Henu ACM Round #13 F】Fibonacci-ish

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 枚举序列的头两个数字是什么 O(N^2) 然后头两个数字确定之后. f[3],f[4]..就确定了 只需查看f[3],f[4]..是 ...

  7. 【Android实战】Android沉浸式状态栏实现(下)

    之前的Android沉浸式状态栏实现并没有考虑软键盘的影响,接下来的内容将会针对这个问题给出解决方式,先看一下效果图 这个是一个留言板的效果图: 即弹出软键盘的时候并不会导致整个布局上移. 详细怎样实 ...

  8. PHP CLI模式下的多进程应用分析

    PHP在非常多时候不适合做常驻的SHELL进程, 他没有专门的gc例程, 也没有有效的内存管理途径. 所以假设用PHP做常驻SHELL, 你会常常被内存耗尽导致abort而unhappy 并且, 假设 ...

  9. [python]CompressionError: bz2 module is not available

    事情是这种,在centos6 上本来是python2.6 然后我下载了一个python2.7.5 安装之后,把默认python改动为python2.7.5版本号. 使用pip安装twisted的时候出 ...

  10. 第一天,Mysql安装,DDL(数据库定义语言),DBA,DML(数据库操纵语言),导入外面的sql文件

    把“D:\mysql-5.6.22-winx64\bin”添加到系统环境变量path中了,然后在任意目录可访问mysql等命令,这样如登录等操作就不需要进入MySQL安装目录才好执行! MySQL下载 ...