java使用默认线程池踩过的坑(二)

云智慧(北京)科技有限公司 陈鑫

是的。一个线程不可以启动两次。那么它是怎么推断的呢?

public synchronized void start() {

/**

* A zero status valuecorresponds to state “NEW”. 0相应的是state NEW

*/

if (threadStatus!= 0) //假设不是NEW state,就直接抛出异常!

throw newIllegalThreadStateException();

group.add(this);

boolean started = false;

try {

start0(); // 启动线程的native方法

started = true;

} finally {

try {

if (!started) {

group.threadStartFailed(this);

}

} catch(Throwable ignore) {

}

}

}

恩,仅仅有是NEW状态才可以调用native方法启动一个线程。好吧。到这里了。就普及也自补一下jvm里的线程状态:

全部的线程状态::

l NEW —— 还没有启动过

l RUNNABLE —— 正在jvm上执行着

l BLOCKED —— 正在等待锁/信号量被释放

l WAITING —— 等待其它某个线程的某个特定动作

l TIMED_WAITING —— A thread that iswaiting for another thread to perform an action for up to a specified waitingtime is in this state.

l TERMINATED —— 退出,停止

线程在某个时间点上仅仅可能存在一种状态。这些状态是jvm里的,并不反映操作系统线程的状态。查一下Thread的API,没有对其状态进行改动的API。那么这条路是不通的吗?

细致考虑一下……

假设把任务做成Runnable实现类,然后在把这个实现类丢进线程池调度器之前,利用此Runnable构造一个Thread,是不是这个Thread对象就行控制这个runnable对象。进而控制在线程池中执行着的task了呢?非也!让我们看看Thread和ThreadPoolExecutor对Runnable的处理吧。

Thread

/* What will berun. */

private Runnabletarget;

结合上面的start()方法,非常easy猜出,start0()会把target弄成一个线程来进行执行。

ThreadPoolExecutor

public void execute(Runnable command){

if (command== null)

thrownew NullPointerException();

int c =ctl.get();

if(workerCountOf(c) < corePoolSize) {

if (addWorker(command, true))

return;

c =ctl.get();

}

if(isRunning(c) && workQueue.offer(command)) {

intrecheck = ctl.get();

if (!isRunning(recheck) && remove(command))

reject(command);

else if(workerCountOf(recheck) == 0)

addWorker(null, false);

}

else if (!addWorker(command, false))

reject(command);

}

private boolean addWorker(RunnablefirstTask, boolean core) {

booleanworkerStarted = false;

booleanworkerAdded = false;

Worker w =null;

try {

finalReentrantLock mainLock = this.mainLock;

w = newWorker(firstTask);

finalThread t = w.thread;

if (t!= null) {

mainLock.lock();

try{

int c = ctl.get();

int rs = runStateOf(c);

               if (rs < SHUTDOWN ||
                   (rs == SHUTDOWN && firstTask == null)) {
                   if (t.isAlive()) // precheck that t is startable
                        throw newIllegalThreadStateException();

workers.add(w);

int s = workers.size();

if (s > largestPoolSize)

largestPoolSize =s;

workerAdded = true;

}

}finally {

mainLock.unlock();

}

if(workerAdded) {

t.start();

workerStarted = true;

}

}

} finally {

if (!workerStarted)

addWorkerFailed(w);

}

return workerStarted;

}

那么Worker又是如何的呢?

Worker

private final class Worker

extendsAbstractQueuedSynchronizer

implementsRunnable

{

finalThread thread;

RunnablefirstTask;

volatilelong completedTasks;

Worker(Runnable firstTask) {

setState(-1); //调用runWorker之前不可以interrupt

this.firstTask = firstTask;

this.thread = getThreadFactory().newThread(this);

}

public voidrun() {

runWorker(this);

}

……

…….

voidinterruptIfStarted() {

Threadt;

if(getState() >= 0 && (t = thread) != null &&!t.isInterrupted()) {

try{

t.interrupt();

}catch (SecurityException ignore) {

}

}

}

}

可见worker里既包装了Runnable对象——task,又包装了一个Thread对象——以自己作为初始化參数。由于worker也是Runnable对象。

然后对外提供了执行与停止接口,run()和interruptIfStarted()。回想上面使用Thread的样例不禁有了新的领悟,我们把一个Thread对象交给ThreadPoolExecutor执行后。实际的调用是对Thread(FileTask())对象,我们临时称之为workerWrapper。

那么我们在池外进行FileTask.interrupt()操作影响的是FileTask对象。而不是workerWrapper。所以可能上面对于start()方法二次调用不是特别适当。更恰当的应该是在fileTask.interrupt()的时候就跑出异常,由于从来没有对fileTask对象执行过start()方法。这时候去interrupt就会出现错误。详细例如以下图:

分析到此,我们已经明白除了调用ThreadPoolExecutor了的interruptWorkers()方法别无其它途径操作这些worker了。

private void interruptWorkers() {

finalReentrantLock mainLock = this.mainLock;

mainLock.lock();

try {

for(Worker w : workers)

w.interruptIfStarted();

} finally {

mainLock.unlock();

}

}

时间: 2024-10-06 07:23:22

java使用默认线程池踩过的坑(二)的相关文章

java使用默认线程池踩过的坑(一)

云智慧(北京)科技有限公司 陈鑫 场景 一个调度器,两个调度任务,分别处理两个目录下的txt文件,某个调度任务应对某些复杂问题的时候会持续特别长的时间,甚至有一直阻塞的可能.我们需要一个manager来管理这些task,当这个task的上一次执行时间距离现在超过5个调度周期的时候,就直接停掉这个线程,然后再重启它,保证两个目标目录下没有待处理的txt文件堆积. 问题 直接使用java默认的线程池调度task1和task2.由于外部txt的种种不可控原因,导致task2线程阻塞.现象就是task1

java使用默认线程池踩过的坑(三)

云智慧(北京)科技有限公司 陈鑫 重新启动线程池 TaskManager public class TaskManager implements Runnable { -.. public TaskManager (Setrunners) { super(); this.runners = runners; executeTasks(runners); } private voidexecuteTasks(Set<FileTask> runners) { for (FileTask task

Java四种线程池newCachedThreadPool,newFixedThreadPool,newScheduledThreadPool,newSingleThreadExecutor

介绍new Thread的弊端及Java四种线程池的使用,对Android同样适用.本文是基础篇,后面会分享下线程池一些高级功能. 1.new Thread的弊端 执行一个异步任务你还只是如下new Thread吗? Java new Thread(new Runnable() { @Override public void run() { // TODO Auto-generated method stub } }).start(); 1 2 3 4 5 6 7 new Thread(new

Java四种线程池

Java四种线程池newCachedThreadPool,newFixedThreadPool,newScheduledThreadPool,newSingleThreadExecutor 时间:2015-10-20 22:37:40      阅读:8762      评论:0      收藏:0      [点我收藏+] 介绍new Thread的弊端及Java四种线程池的使用,对Android同样适用.本文是基础篇,后面会分享下线程池一些高级功能. 1.new Thread的弊端执行一个异

Java中的线程池

综述 在我们的开发中经常会使用到多线程.例如在Android中,由于主线程的诸多限制,像网络请求等一些耗时的操作我们必须在子线程中运行.我们往往会通过new Thread来开启一个子线程,待子线程操作完成以后通过Handler切换到主线程中运行.这么以来我们无法管理我们所创建的子线程,并且无限制的创建子线程,它们相互之间竞争,很有可能由于占用过多资源而导致死机或者OOM.所以在Java中为我们提供了线程池来管理我们所创建的线程. 线程池的使用 采用线程池的好处 在这里我们首先来说一下采用线程池的

Java多线程和线程池(转)

1.为什么要使用线程池 在java中,如果每个请求到达就创建一个新线程,开销是相当大的.在实际使用中,服务器在创建和销毁线程上花费的时间和消耗的系统资源都相当大,甚至可能要比在处理实际的用户请求的时间和资源要多的多.除了创建和销毁线程的开销之外,活动的线程也需要消耗系统资源.如果在一个jvm里创建太多的线程,可能会使系统由于过度消耗内存或“切换过度”而导致系统资源不足.为了防止资源不足,服务器应用程序需要采取一些办法来限制任何给定时刻处理的请求数目,尽可能减少创建和销毁线程的次数,特别是一些资源

Java四种线程池的学习与总结

在Java开发中,有时遇到多线程的开发时,直接使用Thread操作,对程序的性能和维护上都是一个问题,使用Java提供的线程池来操作可以很好的解决问题. 一.new Thread的弊端 执行一个异步任务你还只是如下new Thread吗? new Thread(new Runnable(){ @Override public void run(){ //TODO Auto-generatedmethod stub } } ).start(); 那你就out太多了,new Thread的弊端如下:

Java并发之——线程池

一. 线程池介绍 1.1 简介 线程池是一种多线程处理形式,处理过程中将任务添加到队列,然后在创建线程后自动启动这些任务.线程池的基本思想还是一种对象池的思想,开辟一块内存空间,里面存放了众多(未死亡)的线程,池中线程执行调度由池管理器来处理.当有线程任务时,从池中取一个,执行完成后线程对象归池,这样可以避免反复创建线程对象所带来的性能开销,节省了系统的资源. 多线程技术主要解决处理器单元内多个线程执行的问题,它可以显著减少处理器单元的闲置时间,增加处理器单元的吞吐能力. 假设一个服务器完成一项

Java 四种线程池newCachedThreadPool,newFixedThreadPool,newScheduledThreadPool,newSingleThreadExecutor

原文:http://www.cnblogs.com/zhujiabin/p/5404771.html 介绍new Thread的弊端及Java四种线程池的使用,对Android同样适用.本文是基础篇,后面会分享下线程池一些高级功能. 1.new Thread的弊端执行一个异步任务你还只是如下new Thread吗? new Thread(new Runnable() { @Override public void run() { // TODO Auto-generated method stu