京东二面:线程池中的线程抛出了异常,该如何处理?大部分人都会答错! ...

打印 上一主题 下一主题

主题 1026|帖子 1026|积分 3078

在实际开发中,我们常常会用到线程池,但任务一旦提交到线程池之后,如果发生异常之后,怎么处理? 怎么获取到异常信息?
在了解这个问题之前,可以先看一下 线程池的源码解析,从源码中我们知道了线程池的提交方式:submit和execute的区别,接下来分别使用他们执行带有异常的任务!看结果是怎么样的!
我们先用伪代码模拟一下线程池抛异常的场景:
  1. public class ThreadPoolException {
  2.     public static void main(String[] args) {
  3.         //创建一个线程池
  4.         ExecutorService executorService= Executors.newFixedThreadPool(1);
  5.         //当线程池抛出异常后 submit无提示,其他线程继续执行
  6.         executorService.submit(new task());
  7.         //当线程池抛出异常后 execute抛出异常,其他线程继续执行新任务
  8.         executorService.execute(new task());
  9.     }
  10. }
  11. //任务类
  12. class task implements  Runnable{
  13.     @Override
  14.     public void run() {
  15.         System.out.println("进入了task方法!!!");
  16.         int i=1/0;
  17.     }
  18. }
复制代码
运行结果:

可以看到:submit不打印异常信息,而execute则会打印异常信息!,submit的方式不打印异常信息,显然在生产中,是不可行的,因为我们无法保证线程中的任务永不异常,而如果使用submit的方式出现了异常,直接如上写法,我们将无法获取到异常信息,做出对应的判断和处理,所以下一步需要知道如何获取线程池抛出的异常!
submit()想要获取异常信息就必须使用get()方法!!
  1. //当线程池抛出异常后 submit无提示,其他线程继续执行
  2. Future<?> submit = executorService.submit(new task());
  3. submit.get();
复制代码
submit打印异常信息如下:

推荐一个开源免费的 Spring Boot 最全教程:
https://github.com/javastacks/spring-boot-best-practice
方案一

使用 try -catch
  1. public class ThreadPoolException {
  2.     public static void main(String[] args) {
  3.         //创建一个线程池
  4.         ExecutorService executorService = Executors.newFixedThreadPool(1);
  5.         //当线程池抛出异常后 submit无提示,其他线程继续执行
  6.         executorService.submit(new task());
  7.         //当线程池抛出异常后 execute抛出异常,其他线程继续执行新任务
  8.         executorService.execute(new task());
  9.     }
  10. }
  11. // 任务类
  12. class task implements Runnable {
  13.     @Override
  14.     public void run() {
  15.         try {
  16.             System.out.println("进入了task方法!!!");
  17.             int i = 1 / 0;
  18.         } catch (Exception e) {
  19.             System.out.println("使用了try -catch 捕获异常" + e);
  20.         }
  21.     }
  22. }
复制代码
打印结果:

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

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

UncaughtExceptionHandler 是Thread类一个内部类,也是一个函数式接口。
内部的uncaughtException是一个处理线程内发生的异常的方法,参数为线程对象t和异常对象e。

应用在线程池中如下所示:重写它的线程工厂方法,在线程工厂创建线程的时候,都赋予UncaughtExceptionHandler处理器对象。
  1. public class ThreadPoolException {
  2.     public static void main(String[] args) throws InterruptedException {
  3.         //1.实现一个自己的线程池工厂
  4.         ThreadFactory factory = (Runnable r) -> {
  5.             //创建一个线程
  6.             Thread t = new Thread(r);
  7.             //给创建的线程设置UncaughtExceptionHandler对象 里面实现异常的默认逻辑
  8.             t.setDefaultUncaughtExceptionHandler((Thread thread1, Throwable e) -> {
  9.                 System.out.println("线程工厂设置的exceptionHandler" + e.getMessage());
  10.             });
  11.             return t;
  12.         };
  13.         //2.创建一个自己定义的线程池,使用自己定义的线程工厂
  14.         ExecutorService executorService = new ThreadPoolExecutor(
  15.                 1,
  16.                 1,
  17.                 0,
  18.                 TimeUnit.MILLISECONDS,
  19.                 new LinkedBlockingQueue(10),
  20.                 factory);
  21.         // submit无提示
  22.         executorService.submit(new task());
  23.         Thread.sleep(1000);
  24.         System.out.println("==================为检验打印结果,1秒后执行execute方法");
  25.         // execute 方法被线程工厂factory 的UncaughtExceptionHandler捕捉到异常
  26.         executorService.execute(new task());
  27.     }
  28. }
  29. class task implements Runnable {
  30.     @Override
  31.     public void run() {
  32.         System.out.println("进入了task方法!!!");
  33.         int i = 1 / 0;
  34.     }
  35. }
复制代码
打印结果如下:

根据打印结果我们看到,execute方法被线程工厂factory中设置的 UncaughtExceptionHandler捕捉到异常,而submit方法却没有任何反应!说明UncaughtExceptionHandler在submit中并没有被调用。这是为什么呢?
在日常使用中,我们知道,execute和submit最大的区别就是execute没有返回值,submit有返回值。submit返回的是一个future ,可以通过这个future取到线程执行的结果或者异常信息。
  1. Future<?> submit = executorService.submit(new task());
  2. //打印异常结果
  3.   System.out.println(submit.get());
复制代码

从结果看出:submit并不是丢失了异常,使用future.get()还是有异常打印的!!那为什么线程工厂factory 的UncaughtExceptionHandler没有打印异常呢?猜测是submit方法内部已经捕获了异常, 只是没有打印出来,也因为异常已经被捕获,因此jvm也就不会去调用Thread的UncaughtExceptionHandler去处理异常。
接下来,验证猜想。submit源码在底层还是调用的execute方法,只不过多一层Future封装,并返回了这个Future,这也解释了为什么submit会有返回值
  1. //submit()方法
  2. public <T> Future<T> submit(Callable<T> task) {
  3.      if (task == null) throw new NullPointerException();
  4.      //execute内部执行这个对象内部的逻辑,然后将结果或者异常 set到这个ftask里面
  5.      RunnableFuture<T> ftask = newTaskFor(task);
  6.      // 执行execute方法
  7.      execute(ftask);
  8.      //返回这个ftask
  9.      return ftask;
  10. }
复制代码
可以看到submit也是调用的execute,在execute方法中,我们的任务被提交到了addWorker(command, true) ,然后为每一个任务创建一个Worker去处理这个线程,这个Worker也是一个线程,执行任务时调用的就是Worker的run方法!run方法内部又调用了runworker方法!如下所示:
  1. public void run() {
  2.         runWorker(this);
  3. }
  4. final void runWorker(Worker w) {
  5.      Thread wt = Thread.currentThread();
  6.      Runnable task = w.firstTask;
  7.      w.firstTask = null;
  8.      w.unlock(); // allow interrupts
  9.      boolean completedAbruptly = true;
  10.      try {
  11.       //这里就是线程可以重用的原因,循环+条件判断,不断从队列中取任务
  12.       //还有一个问题就是非核心线程的超时删除是怎么解决的
  13.       //主要就是getTask方法()见下文③
  14.          while (task != null || (task = getTask()) != null) {
  15.              w.lock();
  16.              if ((runStateAtLeast(ctl.get(), STOP) ||
  17.                   (Thread.interrupted() &&
  18.                    runStateAtLeast(ctl.get(), STOP))) &&
  19.                  !wt.isInterrupted())
  20.                  wt.interrupt();
  21.              try {
  22.                  beforeExecute(wt, task);
  23.                  Throwable thrown = null;
  24.                  try {
  25.                   //执行线程
  26.                      task.run();
  27.                      //异常处理
  28.                  } catch (RuntimeException x) {
  29.                      thrown = x; throw x;
  30.                  } catch (Error x) {
  31.                      thrown = x; throw x;
  32.                  } catch (Throwable x) {
  33.                      thrown = x; throw new Error(x);
  34.                  } finally {
  35.                   //execute的方式可以重写此方法处理异常
  36.                      afterExecute(task, thrown);
  37.                  }
  38.              } finally {
  39.                  task = null;
  40.                  w.completedTasks++;
  41.                  w.unlock();
  42.              }
  43.          }
  44.          //出现异常时completedAbruptly不会被修改为false
  45.          completedAbruptly = false;
  46.      } finally {
  47.       //如果如果completedAbruptly值为true,则出现异常,则添加新的Worker处理后边的线程
  48.          processWorkerExit(w, completedAbruptly);
  49.      }
  50. }
复制代码
核心就在 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);里面
  1. public void run() {
  2.      if (state != NEW ||
  3.          !UNSAFE.compareAndSwapObject(this, runnerOffset,
  4.                                       null, Thread.currentThread()))
  5.          return;
  6.      try {
  7.          Callable<V> c = callable;
  8.          if (c != null && state == NEW) {
  9.              V result;
  10.              boolean ran;
  11.              try {
  12.                  result = c.call();
  13.                  ran = true;
  14.              } catch (Throwable ex) {
  15.                  result = null;
  16.                  ran = false;
  17.                  //在此方法中设置了异常信息
  18.                  setException(ex);
  19.              }
  20.              if (ran)
  21.                  set(result);
  22.          }
  23.          //省略下文
  24. 。。。。。。
  25. setException(ex)`方法如下:将异常对象赋予`outcome
  26. protected void setException(Throwable t) {
  27.        if (UNSAFE.compareAndSwapInt(this, stateOffset, NEW, COMPLETING)) {
  28.         //将异常对象赋予outcome,记住这个outcome,
  29.            outcome = t;
  30.            UNSAFE.putOrderedInt(this, stateOffset, EXCEPTIONAL); // final state
  31.            finishCompletion();
  32.        }
  33.    }
复制代码
将异常对象赋予outcome有什么用呢?这个outcome是什么呢?当我们使用submit返回Future对象,并使用Future.get()时, 会调用内部的report方法!
  1. public V get() throws InterruptedException, ExecutionException {
  2.     int s = state;
  3.     if (s <= COMPLETING)
  4.         s = awaitDone(false, 0L);
  5.     //注意这个方法
  6.     return report(s);
  7. }
复制代码
因此,在用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) 方法
  1. private V report(int s) throws ExecutionException {
  2. //设置`outcome`
  3.     Object x = outcome;
  4.     if (s == NORMAL)
  5.      //返回`outcome`
  6.         return (V)x;
  7.     if (s >= CANCELLED)
  8.         throw new CancellationException();
  9.     throw new ExecutionException((Throwable)x);
  10. }
复制代码
重写afterExecute处理execute提交的异常
  1. final void runWorker(Worker w) {
  2. //当前线程
  3.         Thread wt = Thread.currentThread();
  4.         //我们的提交的任务
  5.         Runnable task = w.firstTask;
  6.         w.firstTask = null;
  7.         w.unlock(); // allow interrupts
  8.         boolean completedAbruptly = true;
  9.         try {
  10.             while (task != null || (task = getTask()) != null) {
  11.                 w.lock();
  12.                 if ((runStateAtLeast(ctl.get(), STOP) ||
  13.                      (Thread.interrupted() &&
  14.                       runStateAtLeast(ctl.get(), STOP))) &&
  15.                     !wt.isInterrupted())
  16.                     wt.interrupt();
  17.                 try {
  18.                     beforeExecute(wt, task);
  19.                     Throwable thrown = null;
  20.                     try {
  21.                     //直接就调用了task的run方法
  22.                         task.run(); //如果是futuretask的run,里面是吞掉了异常,不会有异常抛出,
  23.                        // 因此Throwable thrown = null;  也不会进入到catch里面
  24.                     } catch (RuntimeException x) {
  25.                         thrown = x; throw x;
  26.                     } catch (Error x) {
  27.                         thrown = x; throw x;
  28.                     } catch (Throwable x) {
  29.                         thrown = x; throw new Error(x);
  30.                     } finally {
  31.                     //调用线程池的afterExecute方法 传入了task和异常
  32.                         afterExecute(task, thrown);
  33.                     }
  34.                 } finally {
  35.                     task = null;
  36.                     w.completedTasks++;
  37.                     w.unlock();
  38.                 }
  39.             }
  40.             completedAbruptly = false;
  41.         } finally {
  42.             processWorkerExit(w, completedAbruptly);
  43.         }
  44.     }
复制代码
执行结果:我们可以在afterExecute方法内部对异常进行处理

如果要用这个afterExecute处理submit提交的异常, 要额外处理。判断Throwable是否是FutureTask,如果是代表是submit提交的异常,代码如下:
  1. public class ThreadPoolException3 {
  2.     public static void main(String[] args) throws InterruptedException, ExecutionException {
  3.         //1.创建一个自己定义的线程池
  4.         ExecutorService executorService = new ThreadPoolExecutor(
  5.                 2,
  6.                 3,
  7.                 0,
  8.                 TimeUnit.MILLISECONDS,
  9.                 new LinkedBlockingQueue(10)
  10.         ) {
  11.             //重写afterExecute方法
  12.             @Override
  13.             protected void afterExecute(Runnable r, Throwable t) {
  14.                 System.out.println("afterExecute里面获取到异常信息,处理异常" + t.getMessage());
  15.             }
  16.         };
  17.         //当线程池抛出异常后 execute
  18.         executorService.execute(new task());
  19.     }
  20. }
  21. class task3 implements Runnable {
  22.     @Override
  23.     public void run() {
  24.         System.out.println("进入了task方法!!!");
  25.         int i = 1 / 0;
  26.     }
  27. }
复制代码
处理结果如下:

可以看到使用重写afterExecute这种方式,既可以处理execute抛出的异常,也可以处理submit抛出的异常。
版权声明:本文为CSDN博主「知识分子_」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。原文链接:https://blog.csdn.net/qq_45076180/article/details/114552567
近期热文推荐:
1.1,000+ 道 Java面试题及答案整理(2022最新版)
2.劲爆!Java 协程要来了。。。
3.Spring Boot 2.x 教程,太全了!
4.别再写满屏的爆爆爆炸类了,试试装饰器模式,这才是优雅的方式!!
5.《Java开发手册(嵩山版)》最新发布,速速下载!
觉得不错,别忘了随手点赞+转发哦!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

道家人

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表