java并发编程之FutureTask

引言

FutureTask实现了接口Future,同Future一样,代表异步计算的结果。当然,FutureTask除了实现Future接口之外,还实现了Runnable接口,所以,FutureTask既可以由Executor来调度执行,也可以由调度线程调用FutureTask.run()直接执行。

FutureTask状态

根据FutureTask的run方法是否被执行以及是否被执行完成,FutureTask有3种状态:

  1. 未启动:run方法被执行前,FutureTask处于未启动状态;

  2. 已启动:run方法被执行的过程中,FutureTask处于已启动状态;

  3. 已完成:run方法执行完成后正常结束,或者被取消,或者是执行过程中抛出异常导致的异常结束,FutureTask处于已完成状态。

FutureTask状态转换

FutureTask状态转换可以总结为下图:


FutureTask状态转换图
  • 当FutureTask处于未启动或者是已启动状态时,此时还未得到线程执行结果,调用FutureTask.get方法会导致线程阻塞;

  • 当FutureTask处于已完成状态时,此时已经得到线程执行结果,调用FutureTask.get方法会立即返回线程执行结果;

  • 当FutureTask处于未启动状态时,调用FutureTask.cancel方法将会导致该task永远不会被执行;

  • 当FutureTask处于启动状态时,调用FutureTask.cancel方法将会中断该任务的执行,至于会不会对任务产生影响由cancel方法的入参决定;

  • 当FutureTask处于已完成状态,调用FutureTask.cancel方法返回false。

接下来就以run、get和cancel方法为切入点分析FutureTask具体实现。

FutureTask源码分析

在开始分析源码之前,我们先来看看FutureTask的成员变量:


成员变量
  1. state:记录task状态,可取值为0~6;

  2. callable:task实际载体,run方法实际调用callable.call();

  3. outcome:线程执行任务结束后的返回结果;

  4. runner:记录执行task的线程;

  5. waiters:等待task执行结果的线程队列。

构造方法
构造方法

FutureTask提供两个构造方法来封装Callable和Runnable,当构造方法传入参数为Runnable,会通过Executors.callable方法将其转换成Callable。


Executors.callable方法实现
get方法实现

FutureTask提供带超时时间的get和不到超时时间的get:


get方法实现

对比带超时时间和不带超时时间的get方法实现,最为重要的实现就是等待直到task状态变为已完成状态或者等待时间超过超时时间,对应到源码就是加红框的awaitDone方法。接下来我们来具体分析一下awaitDone方法到底是如何来实现线程阻塞等待的。

awaitDone方法实现

awaitDone实现

具体的执行流程如下:

  1. 计算等待时间deadline,如果是带超时时间的get,deadline = 当前时间 + 等待时间,如果是不带超时时间的get,deadline = 0;

  2. 判断线程是否中断,如果线程中断,将当前线程从等待队列waiters中移除,抛出中断异常,否则,跳转到步骤3;

  3. 获取task状态state:

  • 如果task状态为已完成状态,将等待线程节点的线程置为null,返回state;

  • 如果task状态为正在执行,调用Thread.yield()将线程从执行状态变为可执行状态;

  • 否则,跳转到步骤4;

  1. 如果等待线程节点q为null,初始化等待线程节点q,否则,跳转到步骤5;

  2. 如果当前等待线程节点q还未成功进入等待队列waiters,进入线程等待队列,否则,跳转到步骤6;

  3. 判断是否是带超时时间的get:

  • 如果是带超时时间get,判断当前是否超时,如果已经超时,将当前等待节点q从waiters中移出,返回task状态state,如果还未超时,调用LockSupport.parkNanos方法阻塞当前线程;

  • 否则,跳转到步骤7;

  1. 调用LockSupport.park方法,阻塞当前线程,然后跳转到步骤2。

从get方法整个流程可以看出:

  • FutureTask维护一个等待线程队列waiters,如果task还未执行完毕,调用get方法的线程会先进入等待队列自旋等待;

  • awaitDone方法其实是个死循环,直到task状态变为已完成状态或者等待时间超过超时时间或者线程中断才会跳出循环,程序结束;

  • 为了节省开销,线程不会一直自旋等待,而是会阻塞,使用LockSupport的park系列方法实现线程阻塞;

run方法实现
run方法实现

具体执行流程如下:

  1. 判断task状态,如果task还未执行,跳转到步骤2,否则,返回,程序结束;

  2. 通过CAS设置执行task的线程,设置成功,跳转到步骤3,否则,返回,程序结束;

  3. 执行callable.call方法,调用set方法设置call方法返回结果以及task状态;

  4. 设置当前运行当前task的线程为null;

  5. 判断当前task状态,如果task状态为正在中断或者已中断,调用Thread.yield()将线程从执行状态变为可执行状态。

set方法实现

set方法实现

set方法主要干了这两件事:

  1. 设置返回结果outcome以及task状态state;

  2. 调用finishCompletion方法操作等待队列waiters中的等待线程。

finishCompletion实现

finishCompletion实现

整个finishCompletion方法清除和唤醒了等待队列中的等待线程,调用get方法被阻塞的线程也就是在这里调用LockSupport.unpark方法被唤醒的。

cancel方法实现
cancel方法实现
  1. 判断task状态,如果不为未启动状态,返回false,程序结束,否则,跳转到步骤2;

  2. 判断入参mayInterruptIfRunning:

  • true:CSA设置state为正在中断,设置失败返回false,否则中断正在运行task的线程,CAS设置state为已中断;

  • false:CSA设置state为已取消,设置失败返回false,需要注意的是,正在运行task的线程是不会中断的,换句话说,入参为false时不会对task的执行有任何影响。

注:根据代码实现:
- 处于启动状态的task,调用cancel方法是否会对task的执行有所影响完全依赖于cancel方法的入参,true时会有影响,false时不会有影响;
- 处于未启动状态的task,调用cancel方法后,该task将不会再被执行。

  1. 调用finishCompletion方法清除和唤醒等待队列waiters中的等待线程,返回true,程序结束。

从get、run、cancel方法的实现,FutureTask的线程等待与唤醒可以总结为下图:


FutureTask线程等待唤醒

后记

到这里为止,FutureTask的源码就分析就结束了。做一个简短的总结:

  1. FutureTask是通过LockSupport来阻塞线程、唤醒线程;

  2. 对于多线程访问成员变量waiters、state,都采用CAS来操作;

总的来说,FutureTask是一个非常好的CAS和LockSupport搭配使用的例子。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容