线程池的execute方法和submit方法有什么区别?

不羡鸳鸯不羡仙,一行代码调半天。原创:小姐姐味道(微信公众号ID:xjjdog),欢迎分享,转载请保留出处。

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

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

public interface Executor {
void execute(Runnable command);
}
这就是execute方法,接受一个runnable,然后返回为空。也就是说,它接受任务之后,就静悄悄异步去运行了。

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

Future submit(Callable task);
Future submit(Runnable task, T result);
Future<?> submit(Runnable task);
问题
我们尝试运行以下代码:

ExecutorService service = Executors.newFixedThreadPool(1);
Runnable r = () -> System.out.println(1 / 0);
service.submit®;
service.shutdown();
程序静悄悄的什么都没有输出,异常没有,日志也没有,我们的错误直接被吞掉了。

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

Exception in thread “pool-1-thread-1” java.lang.ArithmeticException: / by zero
at com.github.xjjdog.pool.AAA.lambda$main 0 ( A A A . j a v a : 13 ) a t j a v a . u t i l . c o n c u r r e n t . T h r e a d P o o l E x e c u t o r . r u n W o r k e r ( T h r e a d P o o l E x e c u t o r . j a v a : 1149 ) a t j a v a . u t i l . c o n c u r r e n t . T h r e a d P o o l E x e c u t o r 0(AAA.java:13) at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) at java.util.concurrent.ThreadPoolExecutor 0(AAA.java:13)atjava.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)atjava.util.concurrent.ThreadPoolExecutorWorker.run(ThreadPoolExecutor.java:624)
at java.lang.Thread.run(Thread.java:748)
但它也仅仅是输出而已,我们无法使用logback之类的日志框架对其进行记录,因为它这个打印动作我们是不可控的。

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

ExecutorService service = Executors.newFixedThreadPool(1);
Runnable r = () -> System.out.println(1 / 0);
Future f = service.submit®;
f.get();
service.shutdown();
下面是输出结果。

Exception in thread “main” java.util.concurrent.ExecutionException: java.lang.ArithmeticException: / by zero
at java.util.concurrent.FutureTask.report(FutureTask.java:122)
at java.util.concurrent.FutureTask.get(FutureTask.java:192)
at com.github.xjjdog.pool.AAA.main(AAA.java:20)
Caused by: java.lang.ArithmeticException: / by zero
at com.github.xjjdog.pool.AAA.lambda$main 0 ( A A A . j a v a : 16 ) a t j a v a . u t i l . c o n c u r r e n t . E x e c u t o r s 0(AAA.java:16) at java.util.concurrent.Executors 0(AAA.java:16)atjava.util.concurrent.ExecutorsRunnableAdapter.call(Executors.java:511)
at java.util.concurrent.FutureTask.run(FutureTask.java:266)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
at java.lang.Thread.run(Thread.java:748)
但我们平常情况下,使用Future的时候并不多,因为它会阻塞我们的请求。

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

Runnable r = () -> {
System.out.println(“a”);
System.out.println(1 / 0);
};
真是让人恼火啊,想要抛出异常,还是使用execute方便一些。

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

ThreadFactory factory = r->{
Thread thread = Executors.defaultThreadFactory().newThread®;
thread.setUncaughtExceptionHandler( (t,e) -> {
System.out.println(t + “” + e);
e.printStackTrace();//example
});
return thread ;
};

ExecutorService service = Executors.newFixedThreadPool(1,factory);
Runnable r = () -> {
System.out.println(“a”);
System.out.println(1 / 0);
};

service.execute®;
service.shutdown();
运行之后,能够看到我们的自定义异常捕获。

a
Thread[pool-1-thread-1,5,main]java.lang.ArithmeticException: / by zero
End
Java线程池对于异常处理的这些默认行为,以及差别,我是特别抵触的。可以说两种默认行为都很low,我们还需要处理很多动作,才能捕捉到合适的异常。

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

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

后台回复“加群”,带你进入高手如云交流群

推荐阅读:

一图解千愁,jvm内存从来没有这么简单过!
实力解剖一枚挖矿脚本,风骚操作亮瞎双眼
又一P1故障,锅比脸圆
传统企业的人才们,先别忙着跳“互联网”!
面试官很牛,逼我尿遁
又一批长事务,P0故障谁来背锅?
一天有24个小时?别开玩笑了!
《程序人生》杀机!
可怕的“浏览器指纹”,让你在互联网上,无处可藏
2w字长文,让你瞬间拥有「调用链」开发经验
996的乐趣,你是无法想象的
作为高级Java,你应该了解的Linux知识(非广告)
必看!java后端,亮剑诛仙(最全知识点)
学完这100多技术,能当架构师么?(非广告)
Linux上,最常用的一批命令解析(10年精选)
数百篇「原创」文章,助你完成技术「体系化」

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值