Java进阶之ThreadPoolExecutor
作者:网络转载 发布时间:[ 2016/12/19 11:20:57 ] 推荐标签:测试开发技术 Java
到达这一步可能情况为线程池正在关闭或者任务队列已满导致任务提交失败,这里统一进行创建新任务线程处理,创建失败则拒绝任务。
其中,addWorker主要分为两部分逻辑:1)修改状态字以增加任务线程数;2)启动任务线程。代码并不复杂,感兴趣的同学可以看下其实现,这里不再赘述。
任务线程运行逻辑
任务线程(Worker)的主要运行逻辑代码如下:
final void runWorker(Worker w) {
Thread wt = Thread.currentThread();
Runnable task = w.firstTask;
w.firstTask = null;
w.unlock();
boolean completedAbruptly = true;
try {
while (task != null || (task = getTask()) != null) {
w.lock();
if ((runStateAtLeast(ctl.get(), STOP) ||
(Thread.interrupted() &&
runStateAtLeast(ctl.get(), STOP))) &&
!wt.isInterrupted())
wt.interrupt();
try {
beforeExecute(wt, task);
Throwable thrown = null;
try {
task.run();
} catch (RuntimeException x) {
thrown = x; throw x;
} catch (Error x) {
thrown = x; throw x;
} catch (Throwable x) {
thrown = x; throw new Error(x);
} finally {
afterExecute(task, thrown);
}
} finally {
task = null;
w.completedTasks++;
w.unlock();
}
}
completedAbruptly = false;
} finally {
processWorkerExit(w, completedAbruptly);
}
}
这里可以看到,任务线程一直通过 getTask() 来获取任务并执行,直到获取任务失败返回null退出当前线程。
而getTask实现如下:
private Runnable getTask() {
boolean timedOut = false;
for (;;) {
int c = ctl.get();
int rs = runStateOf(c);
if (rs >= SHUTDOWN && (rs >= STOP || workQueue.isEmpty())) {
decrementWorkerCount();
return null;
}
int wc = workerCountOf(c);
boolean timed = allowCoreThreadTimeOut || wc > corePoolSize;
if ((wc > maximumPoolSize || (timed && timedOut))
&& (wc > 1 || workQueue.isEmpty())) {
if (compareAndDecrementWorkerCount(c))
return null;
continue;
}
try {
Runnable r = timed ?
workQueue.poll(keepAliveTime, TimeUnit.NANOSECONDS) :
workQueue.take();
if (r != null)
return r;
timedOut = true;
} catch (InterruptedException retry) {
timedOut = false;
}
}
}
getTask方法返回null时,任务线程会退出。现在我们一步步分析其逻辑:
如果线程池正在关闭则返回null。
当前线程数为如下情形时返回null:1)大于maximumPoolSize;2)大于corePoolSize并且空闲时间超过keepAliveTime;3)设置了allowCoreThreadTimeOut并且当前空闲时间超过keepAliveTime。注意的是,由于存在并发情况,这里做了后一层检查,即如果当前线程为后一个线程且任务队列非空那么会再次获取任务以执行完任务队列里的任务。
通过workQueue.poll()或者workQueue.take()来阻塞获取任务,这两种方法区别在于:前者为等待有间,后者为无限等待。有限等待的情况为,当前任务线程数大于corePoolSize或者设置了allowCoreThreadTimeOut。
总结
本文分析了ThreadPoolExecutor的使用参数以及内部实现,其实现比较简单,感兴趣的同学可以看下其源码:)
本文内容不用于商业目的,如涉及知识产权问题,请权利人联系SPASVO小编(021-61079698-8054),我们将立即处理,马上删除。
相关推荐
Java性能测试有哪些不为众人所知的原则?Java设计模式??装饰者模式谈谈Java中遍历Map的几种方法Java Web入门必知你需要理解的Java反射机制知识总结编写更好的Java单元测试的7个技巧编程常用的几种时间戳转换(java .net 数据库)适合Java开发者学习的Python入门教程Java webdriver如何获取浏览器新窗口中的元素?Java重写与重载(区别与用途)Java变量的分类与初始化JavaScript有这几种测试分类Java有哪四个核心技术?给 Java开发者的10个大数据工具和框架Java中几个常用设计模式汇总java生态圈常用技术框架、开源中间件,系统架构及经典案例等
更新发布
功能测试和接口测试的区别
2023/3/23 14:23:39如何写好测试用例文档
2023/3/22 16:17:39常用的选择回归测试的方式有哪些?
2022/6/14 16:14:27测试流程中需要重点把关几个过程?
2021/10/18 15:37:44性能测试的七种方法
2021/9/17 15:19:29全链路压测优化思路
2021/9/14 15:42:25性能测试流程浅谈
2021/5/28 17:25:47常见的APP性能测试指标
2021/5/8 17:01:11热门文章
常见的移动App Bug??崩溃的测试用例设计如何用Jmeter做压力测试QC使用说明APP压力测试入门教程移动app测试中的主要问题jenkins+testng+ant+webdriver持续集成测试使用JMeter进行HTTP负载测试Selenium 2.0 WebDriver 使用指南