execute和submit的区别与联系
execute和submit都属于线程池的方法,execute只能提交Runnable类型的任务,而submit既能提交Runnable类型任务也能提交Callable类型任务。
execute会直接抛出任务执行时的异常,submit会吃掉异常,可通过Future的get方法将任务执行时的异常重新抛出。
execute所属顶层接口是Executor,submit所属顶层接口是ExecutorService,实现类ThreadPoolExecutor重写了execute方法,抽象类AbstractExecutorService重写了submit方法。
submit和execute由于参数不同有四种实现形式,如下所示,本文主要研究这四种形式在各自使用场景下的区别和联系
-
<T> Future<T> submit(Callable<T> task);
-
<T> Future<T> submit(Runnable task, T result);
-
Future<?> submit(Runnable task);
-
void execute(Runnable command);
关于Runnable和Callable任务如果你还存在疑惑,建议你先看看我的上篇文章Runnable和Callable的区别和联系。
测试代码的整体框架如下:
-
import java.util.concurrent.*;
-
-
public class TestSubmitAndExecute {
-
static ExecutorService executor = Executors.newCachedThreadPool();
-
-
public static void main(String[] args) {
-
initExecutors();
-
/**put test codes here*/
-
-
-
/***/
-
waitToTerminated();
-
-
}
-
-
-
private static void initExecutors() {
-
if (executor.isTerminated()) {
-
executor = Executors.newCachedThreadPool();
-
}
-
}
-
-
private static void waitToTerminated() {
-
executor.shutdown();
-
while (!executor.isTerminated()) {
-
}
-
}
-
-
/**
-
* 测试 submit(Callable<T> task)
-
*
-
* @param callable
-
* @param <T>
-
* @return
-
*/
-
public static <T> T testSubmitCallable(Callable callable) {
-
Future<T> future = executor.submit(callable);
-
T result = null;
-
try {
-
result = future.get();
-
} catch (InterruptedException e) {
-
e.printStackTrace();
-
} catch (ExecutionException e) {
-
e.printStackTrace();
-
}
-
return result;
-
}
-
-
/**
-
* 测试submit(Runnable task, T result)
-
*
-
* @param runnable
-
* @param t
-
* @param <T>
-
* @return
-
*/
-
public static <T> T testSubmitRunnable(Runnable runnable, T t) {
-
Future<T> future = executor.submit(runnable, t);
-
T result = null;
-
try {
-
result = future.get();
-
} catch (InterruptedException e) {
-
e.printStackTrace();
-
} catch (ExecutionException e) {
-
e.printStackTrace();
-
}
-
return result;
-
}
-
-
/**
-
* 测试 submit(Runnable task)
-
* submit提交Runnable任务会默认返回null
-
*
-
* @param runnable
-
* @return
-
*/
-
public static Object testSubmitRunnable(Runnable runnable) {
-
Future<?> future = executor.submit(runnable);
-
Object v = null;
-
try {
-
v = future.get();
-
} catch (InterruptedException e) {
-
e.printStackTrace();
-
} catch (ExecutionException e) {
-
e.printStackTrace();
-
}
-
return v;
-
}
-
-
/**
-
* 测试 execute(Runnable command)
-
* execute会直接抛出异常,submit只有通过调用Future对象的get方法才能获取异常
-
*
-
* @param runnable
-
*/
-
public static void testExecuteRunnable(Runnable runnable) {
-
executor.execute(runnable);
-
}
-
}
这个测试框架提供了4个静态方法用来测试submit和execute总共包含的四种表现形式,除此之外提供initExecutors用于提前检测线程池是否终止,若终止则初始化,waitToTerminated方法用于关闭线程池,并阻塞到线程池终止为止。
除了测试框架之外提供了4个不同的任务,分别测试Callable和Runnable在抛异常时的表现形式。
-
class CallableTask implements Callable<Integer> {
-
@Override
-
public Integer call() throws Exception {
-
int sum = 0;
-
for (int i = 0; i < 520; i++) {
-
sum += i;
-
}
-
return sum;
-
}
-
}
-
-
/**
-
* 会抛异常的CallableTask
-
*/
-
class ExceptionCallableTask implements Callable<Boolean> {
-
public Boolean call() throws Exception {
-
int num = 1 / 0;
-
return false;
-
}
-
}
-
-
class RunnableTask implements Runnable {
-
@Override
-
public void run() {
-
System.out.println("I am a runnable task");
-
}
-
}
-
-
/**
-
* 会抛异常的RunnableTask
-
*/
-
class ExceptionRunableTask implements Runnable {
-
@Override
-
public void run() {
-
int num = 1 / 0;
-
}
-
}
整体结构搭起来,下来就是研究具体差异的时刻了。
1)首先研究Future<?> submit(Runnable task)和void execute(Runnable command),这两个方法都是执行Runnable类型任务,前者有返回值,但是返回值为null,后者无返回值。
-
public static void main(String[] args) {
-
initExecutors();
-
/**put test codes here*/
-
Object object = testSubmitRunnable(new RunnableTask());
-
System.out.println(object);
-
-
testExecuteRunnable(new RunnableTask());
-
-
/***/
-
waitToTerminated();
-
}
很容易观察控制台输出如下:
-
I am a runnable task
-
null
-
I am a runnable task
可以看出submit执行Runnable类型任务时默认返回值为null。如果我们需要submit在提交Runnable任务可以返回非空,就需要用到submit的另外一个重载的方法:<T> Future<T> submit(Runnable task, T result);
2)submit(Runnable task, T result) 方法可以使submit执行完Runnable任务后返回指定的返回值。
main方法如下:
-
public static void main(String[] args) {
-
initExecutors();
-
/**put test codes here*/
-
// Object object = testSubmitRunnable(new RunnableTask());
-
// System.out.println(object);
-
//
-
// testExecuteRunnable(new RunnableTask());
-
-
Integer i = testSubmitRunnable(new RunnableTask(), 3);
-
System.out.println(i);
-
-
Boolean bool = testSubmitRunnable(new RunnableTask(), true);
-
System.out.println(bool);
-
-
String str = testSubmitRunnable(new RunnableTask(), "你好吗");
-
System.out.println(str);
-
-
-
/***/
-
waitToTerminated();
-
}
控制台输出:
-
I am a runnable task
-
3
-
I am a runnable task
-
true
-
I am a runnable task
-
你好吗
可以看出我们输入的什么参数,任务执行完毕后就返回什么参数。
3)submit(Callable<T> task)这个方法没什么好说的,用来提交Callable类型任务,返回值由call方法决定。
main方法如下:
-
public static void main(String[] args) {
-
initExecutors();
-
/**put test codes here*/
-
// Object object = testSubmitRunnable(new RunnableTask());
-
// System.out.println(object);
-
//
-
// testExecuteRunnable(new RunnableTask());
-
-
// Integer i = testSubmitRunnable(new RunnableTask(), 3);
-
// System.out.println(i);
-
//
-
// Boolean bool = testSubmitRunnable(new RunnableTask(), true);
-
// System.out.println(bool);
-
//
-
// String str = testSubmitRunnable(new RunnableTask(), "你好吗");
-
// System.out.println(str);
-
-
Object o = testSubmitCallable(new CallableTask());
-
System.out.println(o);
-
-
/***/
-
waitToTerminated();
-
}
CallableTask的执行逻辑是计算0到520之间的所有整数之和,所以控制台输出:
134940
4)关于execute和submit遭遇异常的表现
execute直接将任务执行时期的异常抛出,main方法和控制台打印分别如下:
-
public static void main(String[] args) {
-
initExecutors();
-
/**put test codes here*/
-
// Object object = testSubmitRunnable(new RunnableTask());
-
// System.out.println(object);
-
//
-
// testExecuteRunnable(new RunnableTask());
-
-
// Integer i = testSubmitRunnable(new RunnableTask(), 3);
-
// System.out.println(i);
-
//
-
// Boolean bool = testSubmitRunnable(new RunnableTask(), true);
-
// System.out.println(bool);
-
//
-
// String str = testSubmitRunnable(new RunnableTask(), "你好吗");
-
// System.out.println(str);
-
-
// Object o = testSubmitCallable(new CallableTask());
-
// System.out.println(o);
-
-
testExecuteRunnable(new ExceptionRunableTask());
-
-
/***/
-
waitToTerminated();
-
}
-
Exception in thread "pool-1-thread-1" java.lang.ArithmeticException: / by zero
-
at ExceptionRunableTask.run(TestRunnableAndCallable.java:38)
-
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
-
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
-
at java.lang.Thread.run(Thread.java:745)
submit比较特殊,如果没有通过Future.get来获取结算结果,则吃掉异常。先将测试方法稍做调整,修改成如下形式:
-
/**
-
* 测试 submit(Callable<T> task)
-
*
-
* @param callable
-
* @param <T>
-
* @return
-
*/
-
public static <T> T testSubmitCallable(Callable callable) {
-
Future<T> future = executor.submit(callable);
-
T result = null;
-
/*
-
try {
-
result = future.get();
-
} catch (InterruptedException e) {
-
e.printStackTrace();
-
} catch (ExecutionException e) {
-
e.printStackTrace();
-
}
-
*/
-
return result;
-
}
当我们在main方法添加如下代码时,控制台其实没有打印任何异常
-
public static void main(String[] args) {
-
initExecutors();
-
/**put test codes here*/
-
// Object object = testSubmitRunnable(new RunnableTask());
-
// System.out.println(object);
-
//
-
// testExecuteRunnable(new RunnableTask());
-
-
// Integer i = testSubmitRunnable(new RunnableTask(), 3);
-
// System.out.println(i);
-
//
-
// Boolean bool = testSubmitRunnable(new RunnableTask(), true);
-
// System.out.println(bool);
-
//
-
// String str = testSubmitRunnable(new RunnableTask(), "你好吗");
-
// System.out.println(str);
-
-
// Object o = testSubmitCallable(new CallableTask());
-
// System.out.println(o);
-
-
// testExecuteRunnable(new ExceptionRunableTask());
-
-
testSubmitCallable(new ExceptionCallableTask());
-
-
/***/
-
waitToTerminated();
-
}
如果将testSubmitCallable代码中被注释的部分取消注释,则可以看到异常信息如下:
-
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 TestSubmitAndExecute.testSubmitCallable(TestSubmitAndExecute.java:58)
-
at TestSubmitAndExecute.main(TestSubmitAndExecute.java:28)
-
Caused by: java.lang.ArithmeticException: / by zero
-
at ExceptionCallableTask.call(TestRunnableAndCallable.java:20)
-
at ExceptionCallableTask.call(TestRunnableAndCallable.java:18)
-
at java.util.concurrent.FutureTask.run(FutureTask.java:266)
-
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
-
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
-
at java.lang.Thread.run(Thread.java:745)
关于execute和submit的简单研究到此结束,谢谢观看。
原文地址:https://www.jianshu.com/p/29610984f1dd
execute和submit的区别与联系的更多相关文章
- 线程池提交任务的两种方式:execute与submit的区别
Java中的线程池在进行任务提交时,有两种方式:execute和submit方法. 一.execute和submit的区别 execute只能提交Runnable类型的任务,无返回值.submit既可 ...
- 线程池 execute 和 submit 的区别
代码示例: public class ThreadPool_Test { public static void main(String[] args) throws InterruptedExcept ...
- ThreadPoolExecutor中execute和submit的区别
1:入参不同 excute() 传入的是 Runable, submit 传入的是 Callable 或 Runable 1):execute 方法源码 public void execute(Run ...
- 线程池中 submit()和 execute()方法有什么区别?(未完成)
线程池中 submit()和 execute()方法有什么区别?(未完成)
- php提交表单时判断 if($_POST[submit])与 if(isset($_POST[submit])) 的区别
if(isset($_POST['submit'])) 它的意思是不是判断是否配置了$_POST['submit'] 这个变量呢?如果有这个变量 在执行其它代码 应该这样用if(isset($_POS ...
- button 和 submit 的区别
表单提交中button和submit的区别submit是button的一个特例,也是button的一种,它把提交这个动作自动集成了,submit和button,二者都以按钮的形式展现,看起来都是按钮, ...
- PL/SQL中直接写SQL语句和用EXECUTE IMMEDIATE方法的区别
PL/SQL中直接写SQL语句和用EXECUTE IMMEDIATE方法的区别 在PL/SQL中在执行SQL语句时可以直接写SQL或者可以把一个SQL语句拼成一个字符串,如下: select * fr ...
- Java线程池中submit()和execute()方法有什么区别
两个方法都可以向线程池提交任务,execute()方法的返回类型是void,它定义在Executor接口中,而submit()方法返回有计算结构的Future对象,它定义在ExecutorServic ...
- Java 线程池中 submit() 和 execute()方法有什么区别?
两个方法都可以向线程池提交任务,execute()方法的返回类型是 void,它定义在 Executor 接口中. 而 submit()方法可以返回持有计算结果的 Future 对象,它定义在 Exe ...
随机推荐
- [Day4] Nginx Http模块一
之前介绍了Nginx作为静态资源服务器的用法,除此之外,Nginx更多的场景是作为反向代理服务器,提高网站的并发和可用性.下面几节着重说一下作为反向代理的http模块,并且了解一些Nginx的架构. ...
- spring boot项目启动报DataSource错误
初建一个简单的spring boot 项目,启动后会报错. Exception encountered during context initialization - cancelling refre ...
- 交叉熵-loss-理解
参考链接: https://blog.csdn.net/tsyccnh/article/details/79163834
- Matlab---length函数
1.length函数:计算向量或矩阵的长度 2.用法说明 y = length(x) 函数计算指定向量或矩阵的长度y.如果参数变量x是向量,则返回其长度:如果参数变量是非空矩阵,则length(x)与 ...
- chown权限命令
chown 命令用途更改与文件关联的所有者或组. 语法chown[ -f ] [ -h] [ -R ] Owner [ :Group ] { File ... | Directory ... } ...
- Luogu P1730 最小密度路径(最短路径+dp)
P1730 最小密度路径 题面 题目描述 给出一张有 \(N\) 个点 \(M\) 条边的加权有向无环图,接下来有 \(Q\) 个询问,每个询问包括 \(2\) 个节点 \(X\) 和 \(Y\) , ...
- Java中的String,StringBuffer和StringBuilder
在了解这个问题的时候查了不少资料,最有帮助的是这个博文:http://swiftlet.net/archives/1694,看了一段时间,咀嚼了一段时间,写一个经过自己消化的博文,希望能帮到大家. 首 ...
- [jnhs]使用netbeans生成的webapp发布到tomcat是需要改名字的,不然就是404Description The origin server did not find a current representation for the target resource or is not willing to disclose that one exists.
2018-12-21更新 退出tomcat然后删除解压之后的文件夹,然后再启动tomcat也可以解决(安装版tomcat) 2018-12-9更新 有时候这样也可以解决 第一次使用tomcat发布we ...
- Dockerfile Tomcat镜像制作
FROM centos MAINTAINER taohaijun "thjtao@126.com" WORKDIR /home #上传安装包 COPY jdk-8u131-linu ...
- free内存监控
语 法: free [-bkmotV][-s <间隔秒数>] 补充说明:free指令会显示内存的使用情况,包括实体内存,虚拟的交换文件内存,共享内存区段,以及系统核心使用的缓冲区等. 参 ...