问题:
1、线程池中submit() 和execute() 方法有什么区别?
2、一个简单的线程池Demo:Runnable
+ThreadPoolExecutor
?
向线程池提交任务的两种方式:
方式一:调用execute()方法,例如:
// Executor接口中的方法
public interface Executor {
void execute(Runnable command);
}
方式二:调用submit()方法,例如:
// ExecutorService接口中的方法
public interface ExecutorService extends Executor {
<T> Future<T> submit(Callable<T> task);
<T> Future<T> submit(Runnable task, T result);
Future<?> submit(Runnable task);
}
以上的submit()和execute()两类方法的区别在哪里呢?大致有以下三点:
1. 二者的接收参数不一样
Execute()方法只能接收Runnable类型的参数,而submit()方法可以接收Callable、Runnable两种类型的参数。Callable类型的任务是可以返回执行结果的,而Runnable类型的任务不可以返回执行结果。
Callable是JDK 1.5加入的执行目标接口,作为Runnable的一种补充,允许有返回值,允许抛出异常。Runnable和Callable的主要区别为:Callable允许有返回值,Runnable不允许有返回值;Runnable不允许抛出异常,Callable允许抛出异常。
2. submit()提交任务后会有返回值,而execute()没有
execute()方法主要用于启动任务的执行,而任务的执行结果和可能的异常调用者并不关心。submit()方法也用于启动任务的执行,但是启动之后会返回Future对象,代表一个异步执行实例,可以通过该异步执行实例去获取结果。
submit()方法自身并不会传递结果,而是返回一个Future异步执行实例,处理过程的结果被包装到Future实例中,调用者可以通过Future.get()方法获取异步执行的结果。通过submit()返回的Future对象获取异步执行结果,演示代码如下:
public class TestSubmit {
@Test
public void test(){
// 向线程池中创建2个线程
ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(2);
// 向线程池中提交2个任务
Future<Integer> future = scheduledThreadPool.submit(new Callable<Integer>() {
@Override
public Integer call() throws Exception {
return RandomUtil.randomInt();
}
});
try {
System.out.println("线程的执行结果"+future.get());// 线程的执行结果1844368679
} catch (InterruptedException e) {
System.out.println("异常调用被中断");
e.printStackTrace();
} catch (ExecutionException e) {
System.out.println("异常调用过程中发生了子异常");
e.printStackTrace();
}
scheduledThreadPool.shutdown();
}
}
3. submit()方便Exception处理
execute()方法在启动任务执行后,任务执行过程中可能发生的异常调用者并不关心。而通过submit()方法返回的Future对象(异步执行实例),可以进行异步执行过程中的异常捕获。
submit()方法自身并不会传递异常,处理过程中的异常都被包装到Future实例中,调用者在调用Future.get()方法获取执行结果时,可以捕获异步执行过程中抛出的受检异常和运行时异常,并进行对应的业务处理。
示例1:
public class TestSubmit {
@Test
public void test(){
// 向线程池中创建2个线程
ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(2);
// 向线程池中提交2个任务
Future<Integer> future = scheduledThreadPool.submit(new Callable<Integer>() {
@Override
public Integer call() throws Exception {
// 抛出NumberFormatException异常
return Integer.valueOf("abc");
}
});
try {
System.out.println("线程的执行结果"+future.get());// 线程的执行结果1844368679
} catch (InterruptedException e) {
System.out.println("异常调用被中断");
e.printStackTrace();
} catch (ExecutionException e) {
System.out.println("异常调用过程中发生了子异常");
e.printStackTrace();
}
scheduledThreadPool.shutdown();
}
}
示例2:
public class TestSubmit {
@Test
public void test1(){
// 向线程池中创建2个线程
ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(2);
// 向线程池中提交2个任务
Future<?> future = scheduledThreadPool.submit(new Runnable() {
@Override
public void run() {
throw new RuntimeException("线程体内抛出了异常");
}
});
try {
// 如果future的返回值为null ,那么任务完成
if(future.get()==null){
System.out.println("任务完成");
}
} catch (Exception e) {
System.out.println(e.getCause().getMessage());
}
scheduledThreadPool.shutdown();
}
}
在ThreadPoolExecutor类的实现中,内部核心的任务提交方法是execute()方法,虽然用户程序通过submit()也可以提交任务,但是实际上submit()方法中最终调用的还是execute()方法。