0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

线程池中如何获取和处理异常

jf_ro2CN3Fa 来源:芋道源码 作者:芋道源码 2022-10-24 15:44 次阅读

1. 模拟线程池抛异常

2. 如何获取和处理异常

1. 模拟线程池抛异常

在实际开发中,我们常常会用到线程池,但任务一旦提交到线程池之后,如果发生异常之后,怎么处理? 怎么获取到异常信息?在了解这个问题之前,可以先看一下 线程池的源码解析,从链接中我们知道了线程池的提交方式:submit和execute的区别,接下来分别使用他们执行带有异常的任务!看结果是怎么样的!

我们先用伪代码模拟一下线程池抛异常的场景:

publicclassThreadPoolException{
publicstaticvoidmain(String[]args){

//创建一个线程池
ExecutorServiceexecutorService=Executors.newFixedThreadPool(1);

//当线程池抛出异常后submit无提示,其他线程继续执行
executorService.submit(newtask());

//当线程池抛出异常后execute抛出异常,其他线程继续执行新任务
executorService.execute(newtask());
}
}

//任务类
classtaskimplementsRunnable{

@Override
publicvoidrun(){
System.out.println("进入了task方法!!!");
inti=1/0;

}
}

运行结果:

e537a120-51ae-11ed-a3b6-dac502259ad0.png

可以看到:submit不打印异常信息,而execute则会打印异常信息!,submit的方式不打印异常信息,显然在生产中,是不可行的,因为我们无法保证线程中的任务永不异常,而如果使用submit的方式出现了异常,直接如上写法,我们将无法获取到异常信息,做出对应的判断和处理,所以下一步需要知道如何获取线程池抛出的异常!

submit()想要获取异常信息就必须使用get()方法!!

//当线程池抛出异常后submit无提示,其他线程继续执行
Futuresubmit=executorService.submit(newtask());
submit.get();

submit打印异常信息如下:

e55ede16-51ae-11ed-a3b6-dac502259ad0.png

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

项目地址:https://gitee.com/zhijiantianya/ruoyi-vue-pro

视频教程:https://doc.iocoder.cn/video/

2. 如何获取和处理异常

方案一:使用 try -catch

publicclassThreadPoolException{
publicstaticvoidmain(String[]args){

//创建一个线程池
ExecutorServiceexecutorService=Executors.newFixedThreadPool(1);

//当线程池抛出异常后submit无提示,其他线程继续执行
executorService.submit(newtask());

//当线程池抛出异常后execute抛出异常,其他线程继续执行新任务
executorService.execute(newtask());
}
}
//任务类
classtaskimplementsRunnable{
@Override
publicvoidrun(){
try{
System.out.println("进入了task方法!!!");
inti=1/0;
}catch(Exceptione){
System.out.println("使用了try-catch捕获异常"+e);
}
}
}

打印结果:

e577ae8c-51ae-11ed-a3b6-dac502259ad0.png

可以看到 submit 和 execute都清晰易懂的捕获到了异常,可以知道我们的任务出现了问题,而不是消失的无影无踪。

方案二:使用Thread.setDefaultUncaughtExceptionHandler方法捕获异常

方案一中,每一个任务都要加一个try-catch 实在是太麻烦了,而且代码也不好看,那么这样想的话,可以用Thread.setDefaultUncaughtExceptionHandler方法捕获异常

e5944114-51ae-11ed-a3b6-dac502259ad0.png

UncaughtExceptionHandler 是Thread类一个内部类,也是一个函数式接口

内部的uncaughtException是一个处理线程内发生的异常的方法,参数为线程对象t和异常对象e。

e5b7c706-51ae-11ed-a3b6-dac502259ad0.png

应用在线程池中如下所示:重写它的线程工厂方法,在线程工厂创建线程的时候,都赋予UncaughtExceptionHandler处理器对象。

publicclassThreadPoolException{
publicstaticvoidmain(String[]args)throwsInterruptedException{


//1.实现一个自己的线程池工厂
ThreadFactoryfactory=(Runnabler)->{
//创建一个线程
Threadt=newThread(r);
//给创建的线程设置UncaughtExceptionHandler对象里面实现异常的默认逻辑
t.setDefaultUncaughtExceptionHandler((Threadthread1,Throwablee)->{
System.out.println("线程工厂设置的exceptionHandler"+e.getMessage());
});
returnt;
};

//2.创建一个自己定义的线程池,使用自己定义的线程工厂
ExecutorServiceexecutorService=newThreadPoolExecutor(
1,
1,
0,
TimeUnit.MILLISECONDS,
newLinkedBlockingQueue(10),
factory);

//submit无提示
executorService.submit(newtask());

Thread.sleep(1000);
System.out.println("==================为检验打印结果,1秒后执行execute方法");

//execute方法被线程工厂factory的UncaughtExceptionHandler捕捉到异常
executorService.execute(newtask());


}


}

classtaskimplementsRunnable{
@Override
publicvoidrun(){
System.out.println("进入了task方法!!!");
inti=1/0;
}
}

打印结果如下:

e5e77c80-51ae-11ed-a3b6-dac502259ad0.png

根据打印结果我们看到,execute方法被线程工厂factory中设置的 UncaughtExceptionHandler捕捉到异常,而submit方法却没有任何反应!说明UncaughtExceptionHandler在submit中并没有被调用。这是为什么呢?

在日常使用中,我们知道,execute和submit最大的区别就是execute没有返回值,submit有返回值。submit返回的是一个future ,可以通过这个future取到线程执行的结果或者异常信息。

Futuresubmit=executorService.submit(newtask());
//打印异常结果
System.out.println(submit.get());
e60e8488-51ae-11ed-a3b6-dac502259ad0.png

从结果看出:submit并不是丢失了异常,使用future.get()还是有异常打印的!!那为什么线程工厂factory 的UncaughtExceptionHandler没有打印异常呢?猜测是submit方法内部已经捕获了异常, 只是没有打印出来,也因为异常已经被捕获,因此jvm也就不会去调用Thread的UncaughtExceptionHandler去处理异常。

接下来,验证猜想:

首先看一下submit和execute的源码:

execute方法的源码在这博客中写的很详细,点击查看execute源码,在此就不再啰嗦了

https://blog.csdn.net/qq_45076180/article/details/108316340

submit源码在底层还是调用的execute方法,只不过多一层Future封装,并返回了这个Future,这也解释了为什么submit会有返回值

//submit()方法
publicFuturesubmit(Callabletask){
if(task==null)thrownewNullPointerException();

//execute内部执行这个对象内部的逻辑,然后将结果或者异常set到这个ftask里面
RunnableFutureftask=newTaskFor(task);
//执行execute方法
execute(ftask);
//返回这个ftask
returnftask;
}

可以看到submit也是调用的execute,在execute方法中,我们的任务被提交到了addWorker(command, true) ,然后为每一个任务创建一个Worker去处理这个线程,这个Worker也是一个线程,执行任务时调用的就是Worker的run方法!run方法内部又调用了runworker方法!如下所示:

publicvoidrun(){
runWorker(this);
}

finalvoidrunWorker(Workerw){
Threadwt=Thread.currentThread();
Runnabletask=w.firstTask;
w.firstTask=null;
w.unlock();//allowinterrupts
booleancompletedAbruptly=true;
try{
//这里就是线程可以重用的原因,循环+条件判断,不断从队列中取任务
//还有一个问题就是非核心线程的超时删除是怎么解决的
//主要就是getTask方法()见下文③
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);
Throwablethrown=null;
try{
//执行线程
task.run();
//异常处理
}catch(RuntimeExceptionx){
thrown=x;throwx;
}catch(Errorx){
thrown=x;throwx;
}catch(Throwablex){
thrown=x;thrownewError(x);
}finally{
//execute的方式可以重写此方法处理异常
afterExecute(task,thrown);
}
}finally{
task=null;
w.completedTasks++;
w.unlock();
}
}
//出现异常时completedAbruptly不会被修改为false
completedAbruptly=false;
}finally{
//如果如果completedAbruptly值为true,则出现异常,则添加新的Worker处理后边的线程
processWorkerExit(w,completedAbruptly);
}
}

核心就在 task.run(); 这个方法里面了, 期间如果发生异常会被抛出。

如果用execute提交的任务,会被封装成了一个runable任务,然后进去 再被封装成一个worker,最后在worker的run方法里面调用runWoker方法, runWoker方法里面执行任务任务,如果任务出现异常,用try-catch捕获异常往外面抛,我们在最外层使用try-catch捕获到了 runWoker方法中抛出的异常。因此我们在execute中看到了我们的任务的异常信息。

那么为什么submit没有异常信息呢? 因为submit是将任务封装成了一个futureTask ,然后这个futureTask被封装成worker,在woker的run方法里面,最终调用的是futureTask的run方法, 猜测里面是直接吞掉了异常,并没有抛出异常,因此在worker的runWorker方法里面无法捕获到异常。

下面来看一下futureTask的run方法,果不其然,在try-catch中吞掉了异常,将异常放到了 setException(ex);里面

publicvoidrun(){
if(state!=NEW||
!UNSAFE.compareAndSwapObject(this,runnerOffset,
null,Thread.currentThread()))
return;
try{
Callablec=callable;
if(c!=null&&state==NEW){
Vresult;
booleanran;
try{
result=c.call();
ran=true;
}catch(Throwableex){
result=null;
ran=false;
//在此方法中设置了异常信息
setException(ex);
}
if(ran)
set(result);
}
//省略下文
。。。。。。
setException(ex)`方法如下:将异常对象赋予`outcome
protectedvoidsetException(Throwablet){
if(UNSAFE.compareAndSwapInt(this,stateOffset,NEW,COMPLETING)){
//将异常对象赋予outcome,记住这个outcome,
outcome=t;
UNSAFE.putOrderedInt(this,stateOffset,EXCEPTIONAL);//finalstate
finishCompletion();
}
}

将异常对象赋予outcome有什么用呢?这个outcome是什么呢?当我们使用submit返回Future对象,并使用Future.get()时, 会调用内部的report方法!

publicVget()throwsInterruptedException,ExecutionException{
ints=state;
if(s<= COMPLETING)
        s = awaitDone(false, 0L);
    //注意这个方法
    return report(s);
}

reoport里面实际上返回的是outcome ,刚好之前的异常就set到了这个outcome里面

privateVreport(ints)throwsExecutionException{
//设置`outcome`
Objectx=outcome;
if(s==NORMAL)
//返回`outcome`
return(V)x;
if(s>=CANCELLED)
thrownewCancellationException();
thrownewExecutionException((Throwable)x);
}

因此,在用submit提交的时候,runable对象被封装成了future ,future 里面的 run方法在处理异常时, try-catch了所有的异常,通过setException(ex);方法设置到了变量outcome里面, 可以通过future.get获取到outcome。

所以在submit提交的时候,里面发生了异常, 是不会有任何抛出信息的。而通过future.get()可以获取到submit抛出的异常!在submit里面,除了从返回结果里面取到异常之外, 没有其他方法。因此,在不需要返回结果的情况下,最好用execute ,这样就算没有写try-catch,疏漏了异常捕捉,也不至于丢掉异常信息。

方案三:重写afterExecute进行异常处理

通过上述源码分析,在excute的方法里面,可以通过重写afterExecute进行异常处理,但是注意! 这个也只适用于excute提交(submit的方式比较麻烦,下面说),因为submit的task.run里面把异常吞了,根本不会跑出来异常,因此也不会有异常进入到afterExecute里面。

在runWorker里面,调用task.run之后,会调用线程池的 afterExecute(task, thrown) 方法

finalvoidrunWorker(Workerw){
//当前线程
Threadwt=Thread.currentThread();
//我们的提交的任务
Runnabletask=w.firstTask;
w.firstTask=null;
w.unlock();//allowinterrupts
booleancompletedAbruptly=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);
Throwablethrown=null;
try{
//直接就调用了task的run方法
task.run();//如果是futuretask的run,里面是吞掉了异常,不会有异常抛出,
//因此Throwablethrown=null;也不会进入到catch里面
}catch(RuntimeExceptionx){
thrown=x;throwx;
}catch(Errorx){
thrown=x;throwx;
}catch(Throwablex){
thrown=x;thrownewError(x);
}finally{
//调用线程池的afterExecute方法传入了task和异常
afterExecute(task,thrown);
}
}finally{
task=null;
w.completedTasks++;
w.unlock();
}
}
completedAbruptly=false;
}finally{
processWorkerExit(w,completedAbruptly);
}
}

重写afterExecute处理execute提交的异常

publicclassThreadPoolException3{
publicstaticvoidmain(String[]args)throwsInterruptedException,ExecutionException{


//1.创建一个自己定义的线程池
ExecutorServiceexecutorService=newThreadPoolExecutor(
2,
3,
0,
TimeUnit.MILLISECONDS,
newLinkedBlockingQueue(10)
){
//重写afterExecute方法
@Override
protectedvoidafterExecute(Runnabler,Throwablet){
System.out.println("afterExecute里面获取到异常信息,处理异常"+t.getMessage());
}
};

//当线程池抛出异常后execute
executorService.execute(newtask());
}
}

classtask3implementsRunnable{
@Override
publicvoidrun(){
System.out.println("进入了task方法!!!");
inti=1/0;
}
}

执行结果:我们可以在afterExecute方法内部对异常进行处理

e62a175c-51ae-11ed-a3b6-dac502259ad0.png

如果要用这个afterExecute处理submit提交的异常, 要额外处理。判断Throwable是否是FutureTask,如果是代表是submit提交的异常,代码如下:

publicclassThreadPoolException3{
publicstaticvoidmain(String[]args)throwsInterruptedException,ExecutionException{


//1.创建一个自己定义的线程池
ExecutorServiceexecutorService=newThreadPoolExecutor(
2,
3,
0,
TimeUnit.MILLISECONDS,
newLinkedBlockingQueue(10)
){
//重写afterExecute方法
@Override
protectedvoidafterExecute(Runnabler,Throwablet){
//这个是excute提交的时候
if(t!=null){
System.out.println("afterExecute里面获取到excute提交的异常信息,处理异常"+t.getMessage());
}
//如果r的实际类型是FutureTask那么是submit提交的,所以可以在里面get到异常
if(rinstanceofFutureTask){
try{
Futurefuture=(Future)r;
//get获取异常
future.get();

}catch(Exceptione){
System.out.println("afterExecute里面获取到submit提交的异常信息,处理异常"+e);
}
}
}
};
//当线程池抛出异常后execute
executorService.execute(newtask());

//当线程池抛出异常后submit
executorService.submit(newtask());
}
}

classtask3implementsRunnable{
@Override
publicvoidrun(){
System.out.println("进入了task方法!!!");
inti=1/0;
}
}

处理结果如下:

e63f47bc-51ae-11ed-a3b6-dac502259ad0.png

可以看到使用重写afterExecute这种方式,既可以处理execute抛出的异常,也可以处理submit抛出的异常

审核编辑:彭静
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 函数
    +关注

    关注

    3

    文章

    4325

    浏览量

    62552
  • 管理系统
    +关注

    关注

    1

    文章

    2476

    浏览量

    35893
  • 线程池
    +关注

    关注

    0

    文章

    57

    浏览量

    6844

原文标题:线程池中线程抛了异常,该如何处理?

文章出处:【微信号:芋道源码,微信公众号:芋道源码】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    线程获取系统时间

    是vs程序,多线程获取系统时间。
    发表于 07-17 09:09

    memcache主线程和工人线程进行通信的设计实现

    1、memcache多线程模型memcache 是采用单进程多线程模型,内部使用 lib 事件库来处理网络请求。其工作是主线程负责接受的客户端请求,然后轮询模式新任务模式
    发表于 06-23 16:46

    RT-Thread如何去发现该线程已经进入异常的死循环了呢?

    各位大佬,假在某个线程的执行代码的异常执行流中,如else处理中写了while(1)这种死循环代码,rt-thread如何去发现该线程已经进入异常
    发表于 10-18 09:57

    java异常处理的设计与重构

    在程序设计中,进行异常处理是非常关键和重要的一部分。一个程序的异常处理框架的好坏直接影响到整个项目的代码质量以及后期维护成本和难度。试想一下,如果一个项目从头到尾没有考虑过
    发表于 09-27 15:40 1次下载
    java<b class='flag-5'>异常</b><b class='flag-5'>处理</b>的设计与重构

    从电池中获取更多能量

    从电池中获取更多能量
    发表于 04-24 15:45 6次下载
    从电<b class='flag-5'>池中</b><b class='flag-5'>获取</b>更多能量

    原理解析:线程池中多余的线程是如何回收的?

    最近阅读了JDK线程池ThreadPoolExecutor的源码,对线程池执行任务的流程有了大体了解,实际上这个流程也十分通俗易懂,就不再赘述了,别人写的比我好多了。
    的头像 发表于 11-11 09:57 951次阅读

    什么是线程线程池中线程实现复用的原理

    一般建议自定义线程工厂,构建线程的时候设置线程的名称,这样就在查日志的时候就方便知道是哪个线程执行的代码。
    发表于 01-29 13:44 1741次阅读

    线程线程

    线程池通常用于服务器应用程序。 每个传入请求都将分配给线程池中的一个线程,因此可以异步处理请求,而不会占用主
    的头像 发表于 02-28 09:53 782次阅读
    多<b class='flag-5'>线程</b>之<b class='flag-5'>线程</b>池

    Java线程池核心原理

    看过Java线程池源码的小伙伴都知道,在Java线程池中最核心的类就是ThreadPoolExecutor,
    的头像 发表于 04-21 10:24 847次阅读

    核心线程数和最大线程数区别

    核心线程数和最大线程数区别 核心线程数是线程池中一直存在的线程数,不会被回收。最大
    的头像 发表于 06-01 09:33 7647次阅读

    cpu核心数和线程数的关系

    核心线程数是线程池中一直存在的线程数,不会被回收。最大线程数是线程
    的头像 发表于 06-01 17:41 9682次阅读

    线程池的线程怎么释放

    线程分组看,pool名开头线程占616条,而且waiting状态也是616条,这个点就非常可疑了,我断定就是这个pool开头线程池导致的问题。我们先排查为何这个线程
    发表于 07-31 10:49 2269次阅读
    <b class='flag-5'>线程</b>池的<b class='flag-5'>线程</b>怎么释放

    线程事务怎么回滚?一个简单示例演示多线程事务

    在spring中可以使用@Transactional注解去控制事务,使出现异常时会进行回滚,在多线程中,这个注解则不会生效,如果主线程需要先执行一些修改数据库的操作,当子线程在进行
    发表于 08-09 12:22 663次阅读
    多<b class='flag-5'>线程</b>事务怎么回滚?一个简单示例演示多<b class='flag-5'>线程</b>事务

    核心线程数和最大线程数怎么设置

    核心线程数和最大线程数是Java线程池中重要的参数,用来控制线程池中线程的数量和行为。正确地设置
    的头像 发表于 12-01 13:50 9017次阅读

    线程池七大核心参数执行顺序

    线程池是一种用于管理和调度线程执行的技术,通过将任务分配到线程池中线程进行处理,可以有效地控制
    的头像 发表于 12-04 16:45 1031次阅读