线程池的Execute方法和Submit方法有什么区别?

网络 通信技术
文章内容很聚焦,但干货十足。不注意的话你可能会落入陷阱。concurrent包里的ExecutorService,是一个接口,继承的是Executor,而Executor里只有一个方法。

 [[335211]]

本文转载自微信公众号「小姐姐味道」,作者小姐姐养的狗 。转载本文请联系小姐姐味道公众号。

文章内容很聚焦,但干货十足。不注意的话你可能会落入陷阱。

concurrent包里的ExecutorService,是一个接口,继承的是Executor,而Executor里只有一个方法。

  1. public interface Executor { 
  2.     void execute(Runnable command); 

这就是execute方法,接受一个runnable,然后返回为空。也就是说,它接受任务之后,就静悄悄异步去运行了。

我们再来看submit方法。区别就是submit方法,会返回一个Future对象。显然它是比execute方法多了一些内容的。

  1. <T> Future<T> submit(Callable<T> task); 
  2. <T> Future<T> submit(Runnable task, T result); 
  3. Future<?> submit(Runnable task); 

问题

我们尝试运行以下代码:

  1. ExecutorService service = Executors.newFixedThreadPool(1); 
  2. Runnable r = () -> System.out.println(1 / 0); 
  3. service.submit(r); 
  4. service.shutdown(); 

程序静悄悄的什么都没有输出,异常没有,日志也没有,我们的错误直接被吞掉了。

把submit方法换成execute方法,可以看到异常能够正常输出。为了避免抄袭,我还是输出一些自定义的堆栈吧。

  1. Exception in thread "pool-1-thread-1" java.lang.ArithmeticException: / by zero 
  2.     at com.github.xjjdog.pool.AAA.lambda$main$0(AAA.java:13) 
  3.     at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) 
  4.     at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) 
  5.     at java.lang.Thread.run(Thread.java:748) 

但它也仅仅是输出而已,我们无法使用logback之类的日志框架对其进行记录,因为它这个打印动作我们是不可控的。

解决方法

首先看下submit 方式的解决方法。通过返回的Future,执行它的get方法,即可获取完成的错误堆栈。

  1. ExecutorService service = Executors.newFixedThreadPool(1); 
  2. Runnable r = () -> System.out.println(1 / 0); 
  3. Future f = service.submit(r); 
  4. f.get(); 
  5. service.shutdown(); 

下面是输出结果。

  1. Exception in thread "main" java.util.concurrent.ExecutionException: java.lang.ArithmeticException: / by zero 
  2.     at java.util.concurrent.FutureTask.report(FutureTask.java:122) 
  3.     at java.util.concurrent.FutureTask.get(FutureTask.java:192) 
  4.     at com.github.xjjdog.pool.AAA.main(AAA.java:20) 
  5. Caused by: java.lang.ArithmeticException: / by zero 
  6.     at com.github.xjjdog.pool.AAA.lambda$main$0(AAA.java:16) 
  7.     at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511) 
  8.     at java.util.concurrent.FutureTask.run(FutureTask.java:266) 
  9.     at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) 
  10.     at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) 
  11.     at java.lang.Thread.run(Thread.java:748) 

但我们平常情况下,使用Future的时候并不多,因为它会阻塞我们的请求。

你可能怀疑不调用get,我们的代码没有运行,其实不是的。把runnable改成如下代码,不调用get方法,发现程序只输出了一个a。

  1. Runnable r = () -> { 
  2.     System.out.println("a"); 
  3.     System.out.println(1 / 0); 
  4. }; 

真是让人恼火啊,想要抛出异常,还是使用execute方便一些。

但我们上面说到,execute的方式,错误也是无法捕捉。其实我们可以曲线救国的绕一下去解决。解决方式就是使用ThreadFactory,实现它的UncaughtExceptionHandler。具体代码如下:

  1. ThreadFactory factory = r->{ 
  2.     Thread thread = Executors.defaultThreadFactory().newThread(r); 
  3.     thread.setUncaughtExceptionHandler( (t,e) -> { 
  4.         System.out.println(t + "" + e); 
  5.         e.printStackTrace();//example 
  6.     }); 
  7.     return thread ; 
  8. }; 
  9.  
  10. ExecutorService service = Executors.newFixedThreadPool(1,factory); 
  11. Runnable r = () -> { 
  12.     System.out.println("a"); 
  13.     System.out.println(1 / 0); 
  14. }; 
  15.  
  16. service.execute(r); 
  17. service.shutdown(); 

运行之后,能够看到我们的自定义异常捕获。

  1. Thread[pool-1-thread-1,5,main]java.lang.ArithmeticException: / by zero 

EndJava线程池对于异常处理的这些默认行为,以及差别,我是特别抵触的。可以说两种默认行为都很low,我们还需要处理很多动作,才能捕捉到合适的异常。

多线程编程本来就难,又搞出这么两套东西来。找个日志吧,习惯性的往项目的error日志里去找,并没有。真是苦了开发同学。

作者简介:小姐姐味道 (xjjdog),一个不允许程序员走弯路的公众号。聚焦基础架构和Linux。十年架构,日百亿流量,与你探讨高并发世界,给你不一样的味道。我的个人微信xjjdog0,欢迎添加好友,进一步交流。

 

责任编辑:武晓燕 来源: 小姐姐味道
相关推荐

2013-07-30 13:35:12

methodfunction

2022-02-08 07:02:32

进程线程操作系统

2022-01-11 06:53:23

面试重写重载

2022-03-16 07:33:40

守护线程用户线程语言

2020-03-09 20:56:19

LoRaLoRaWAN无线技术

2022-06-06 14:53:02

LoRaLoRaWAN

2022-09-07 18:32:57

并发编程线程

2020-11-09 14:07:53

PyQtQt编程

2022-09-08 18:38:26

LinuxWindowsmacOS

2022-08-31 08:33:54

Bash操作系统Linux

2022-08-02 08:23:37

SessionCookies

2022-02-27 15:33:22

安全CASBSASE

2021-12-17 14:40:02

while(1)for(;;)语言

2021-05-16 14:26:08

RPAIPACIO

2024-03-05 18:59:59

前端开发localhost

2018-07-20 14:00:51

LinuxmacOS内核

2023-12-15 09:21:17

ObjectJavaString

2022-08-22 07:06:32

MyBatisSQL占位符

2020-08-02 23:20:36

JavaScriptmap()forEach()

2022-01-16 07:46:53

SpringDataASSM
点赞
收藏

51CTO技术栈公众号