1、线程执行的任务发生异常,默认线程是把堆栈信息抛到控制台。
2、如果线程设置了UncaughtExceptionHandler,那么这个UncaughtExceptionHandler就会被执行,这个handler只适用于这个线程没有自己处理异常及其他处理异常的Handler。
3、线程抛出异常,这个线程就死了。
4、线程池submit()任务,如果任务抛出异常,这个线程会把异常吃了,并把异常信息赋值给FutureTask的outcome,代码FutureTask:
public void run() {
if (state != NEW ||
!UNSAFE.compareAndSwapObject(this, runnerOffset,
null, Thread.currentThread()))
return;
try {
Callable<V> c = callable;
if (c != null && state == NEW) {
V result;
boolean ran;
try {
result = c.call();
ran = true;
} catch (Throwable ex) {
result = null;
ran = false;
setException(ex);
}
if (ran)
set(result);
}
} finally {
// runner must be non-null until state is settled to
// prevent concurrent calls to run()
runner = null;
// state must be re-read after nulling runner to prevent
// leaked interrupts
int s = state;
if (s >= INTERRUPTING)
handlePossibleCancellationInterrupt(s);
}
}
如果Future不调用get()的话,那么这个异常就永久消失了。
如果调用get()的话,任务中抛出的异常会被封装成ExecutionException(your throwable)返回给你。
5、execute() 会抛异常,要么直接抛到system.err, 要么被你的uncaughtExceptionHandler捕获
6、submit() 永远不会抛异常,除了future.get()。
- 线程池是如何保持core size数目的线程不死?
如果不允许core线程死亡,通过queue.wait()来保存线程阻塞。
7、不要调用Executors工具类来生成线程池,因为用的是无界队列,会OOM
8、coreSize, maxSize和blockingQueue在提交任务时:如果池中线程数小于coreSize,会自动创建新线程执行;池中线程数大于coreSize且小于maxSize,进入队列;队列满了,会创建新线程;池中线程数大于maxSize,走拒绝策略。AbortPolicy:抛异常,DiscardPolicy:不干任何事;DiscardOldestPolicy:抛弃最老的一个任务,新任务进队列;CallerRunPolicy:调用者执行,最常用。
9、防止线程池被强制关闭,一般kill -15 PID,jvm需要注册shutdownHook,调用线程池的shutdown,可以设置awaitTime内关闭。
网友评论