剖析ThreadPoolExecutor 原理

背景

ThreadExecutorPool是使用最多的线程池组件,了解它的原始资料最好是从从设计者(Doug Lea)的口中知道它的来龙去脉。在Jdk1.6中,ThreadPoolExecutor直接继承了AbstractExecutorService, 并层级实现了ExecutorService和Executor接口。

实现原理

当一个任务通过executor调度的时候,线程池需要做:

1.判断当前线程数目是否小于核心线程数目,如果小于,则创建线程调度(注意:即便有空闲线程依旧创建);

2.如果线程线程数目等于核心线程数目,但等待队列未满,则将任务加入等待队列BlockQueue;

3.如果线程数目等于核心线程数,切等待队列慢的话,且当前线程数目小于最大可创建数目,则急救创建新的线程数处理,但是处理完后会销毁这些线程;

4.如果线程数目等于最大可创建线程数,则执行拒绝策略RejectedExecutionHandller;

下面我们看看源码:


源码

ctl.get()方法是判断线程池的状态,目前有以下状态:

RUNNING:接受新任务并处理队列任务

SHUTDOWN:不接受新任务,但处理队列任务

STOP:不接受新任务,不处理排队任务

TIDYING: 所有的线程都终止了(包括queue中),同时workerCount为0,那么此时进入TIDYING

TERMINATED: terminated()方法结束,变为TERMINATED

我们看到如果当前线程数目小于核心线程数的话,就调用addWork,即增加线程数(Worker是一个实现了Runable的内部类,可以直接当做是线程,负责处理任务),否则的话,则直接丢入等待队列,丢入队列时候依旧判断线程状态,如果状态非运行状态,也就是不是上面提到的Running 的状态,则直接拒绝添加任务;

最后如果无法添加线程,则执行拒绝策略;

拒绝的策略有以下几类:

1. Reject 直接抛出Reject exception

2. Discard 直接忽略该runnable,不可取

3. DiscardOldest 丢弃最早入队列的的任务

4. CallsRun 直接让原先的client thread做为worker线程,进行执行,大致意思就是让调用方自己控制

现在我们再看看addWorkder这个核心方法是如何添加的:


源码

首先依旧判断线程状态,因为只有状态是RUNNING才可以创建线程,如果满足状态的怕,则通过调用CAS的方法循环尝试判断是否可以创建线程,如满足则,跳出循环,下面创建Workder 


源码

首先看看Worker这个内部类:

成员变量:

1.Thread 正在运行的线程

2.firstTask  开始运行的初始任务,可以为空

3.completedTasks 线程计数器

Worker执行任务时,即上面的run()方法,需要先获取runLock,此处的目的是在任务的执行过程中防止worker线程被中断。然后双重检查是否线程池已停止或者中断。最好开始执行任务,此时调用用户自定的钩子方法,可在执行前和执行后作相应的处理。在Worker自身的run方法体中,需要先获取任务,调用实际的runTask。在获取任务的操作中,getTask()由于是阻塞获取,则保证了线程的最终存活的可能。

最后我们看看线程池的终止:


源码


cheackShutdownAccess这方法的源码注释大致的意思就是坚持调用该方法时候是否有权限允许中断线程,如果允许则将线程池的状态改成SHUTDOWN,即不在添加任务,但依旧处理队列里的任务,接着是中断interruptIdleWorkers()线程,我们可以看看这方法的源码


源码

循环遍历,尝试获取锁,并中断,刚刚我们看到work的run方法是加锁的,也就是说,如果没有获取到锁的话,是不能中断,也就是说,正在运行的线程是不会被中断的,也就是说针对空闲线程。最后会执行finally的tryTerminate方法也就是进入终止状态,下面我继续看看源码:


源码

进入终止状态的时候 需要由SHUTDOWN进入TIDING而这个状态需要workQueue为空,而进入TERMINATED状态则需要把workcount清0,。

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

推荐阅读更多精彩内容

  • 博客链接:http://www.ideabuffer.cn/2017/04/04/深入理解Java线程池:Thre...
    闪电是只猫阅读 15,815评论 15 133
  • 前言 JDK中为我们提供了一个并发线程框架,它是的我们可以在有异步任务或大量并发任务需要执行时可以使用它提供的线程...
    Justlearn阅读 1,778评论 0 10
  • 前言:线程是稀缺资源,如果被无限制的创建,不仅会消耗系统资源,还会降低系统的稳定性,合理的使用线程池对线程进行统一...
    SDY_0656阅读 703评论 0 1
  • 拖着皱纹的老人 随着船的残骸沉入深海 兴奋的脸上 没有恐惧的影子 他看到海底的宫殿 拿着权杖的众神将他捉住 他期待...
    伍丁零阅读 87评论 0 2
  • 我想看透这个局,但是眼光不够,视线太窄 ————大学教师们的利益
    就这样一个星期天阅读 145评论 0 0