Java 线程池 execute() 和 submit() 对比

execute()submit() 都是线程池提交任务 的核心方法,都属于 ExecutorService 线程池接口,核心区别execute() 无返回值、不处理异常;submit() 有返回值、会捕获异常

下面从核心定义、使用场景、返回值、异常处理、方法归属等维度做清晰对比,附实战代码。


一、核心区别速查表

对比项 execute() submit()
返回值 void无返回结果 Future<T>可获取任务执行结果/状态
异常处理 直接抛出异常(会终止线程,从线程池移除) 自动捕获异常,需通过 Future.get() 才抛出
任务类型 只能提交 Runnable 无返回值任务 可提交 Runnable + Callable 有返回值任务
使用场景 执行不需要结果、不关心异常的异步任务 执行需要结果、需要捕获异常的异步任务
方法归属 定义在 Executor 顶层接口 定义在 ExecutorService 子接口
源码设计 最基础的任务提交方法 基于 execute() 封装,功能更强

二、详细解析

1. 返回值区别(最核心)

  • execute()void,提交后无法跟踪任务是否执行完成、无法获取结果。
  • submit() :返回 Future 对象,可通过它:
    • 判断任务是否完成
    • 取消任务
    • 获取任务执行结果
    • 获取任务抛出的异常

2. 异常处理区别(关键坑点)

  • execute() :任务抛出未捕获异常 时,直接打印堆栈、终止线程,线程会从线程池被移除。
  • submit() :无论任务抛什么异常,都不会直接抛出 ,异常会被存起来,只有调用 Future.get() 时才会抛出。

3. 支持的任务类型区别

  • execute() :仅支持 Runnable(无返回值)。
  • submit() :支持两种任务:
    1. Runnable(无返回值)
    2. Callable<T>有返回值,能抛异常)

三、实战代码示例

1. execute() 使用

java 复制代码
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class ExecuteDemo {
    public static void main(String[] args) {
        ExecutorService pool = Executors.newSingleThreadExecutor();
        
        // 提交无返回值任务
        pool.execute(() -> {
            System.out.println("execute 执行任务");
            // 异常会直接抛出,导致线程死亡
            // int i = 1 / 0;
        });
        
        pool.shutdown();
    }
}

✅ 适用:日志打印、消息发送、异步更新缓存等不需要结果、不关心异常的任务。


2. submit() 使用(获取返回值)

java 复制代码
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;

public class SubmitDemo {
    public static void main(String[] args) throws Exception {
        ExecutorService pool = Executors.newSingleThreadExecutor();

        // 提交 Callable 任务,有返回值
        Future<Integer> future = pool.submit(new Callable<Integer>() {
            @Override
            public Integer call() throws Exception {
                System.out.println("submit 执行任务");
                return 1 + 1; // 返回结果
            }
        });

        // 获取任务结果(阻塞等待)
        Integer result = future.get();
        System.out.println("任务结果:" + result); // 输出 2

        pool.shutdown();
    }
}

✅ 适用:异步计算、接口并行调用、需要获取执行结果/处理异常的任务。


3. submit() 异常处理演示

java 复制代码
Future<Integer> future = pool.submit(() -> {
    System.out.println("submit 任务");
    int i = 1 / 0; // 抛异常
    return 100;
});

// 异常不会直接抛出!
// 只有调用 get() 才会抛出 ExecutionException
try {
    future.get();
} catch (Exception e) {
    System.out.println("捕获到异常:" + e.getCause());
}

四、底层原理(简单理解)

submit() 本质是execute() 的封装

  1. 把任务包装成 FutureTask(同时实现 Runnable + Future
  2. 调用 execute() 提交
  3. 执行结果/异常存入 FutureTask,通过 get() 获取

五、如何选择?

  1. 只用 execute()
    任务不需要返回值异常自己处理(try-catch)、不关心执行状态。
  2. 必须用 submit()
    • 需要获取任务结果
    • 需要判断任务是否完成/取消任务
    • 需要统一捕获任务异常
    • 提交 Callable 带返回值的任务

总结

  1. execute():轻量、无返回、异常直接抛 → 适合简单异步执行。
  2. submit():强大、有返回、可捕获异常 → 适合需要结果/管控的任务。
  3. 日常开发:简单任务用 execute,复杂任务用 submit + Future
相关推荐
AIGS0011 小时前
生产运营三大瓶颈,工业AI怎么破局?
java·人工智能·人工智能ai大模型应用
方也_arkling1 小时前
【Java-Day19】集合1(Collect单列集合)
java·开发语言
Xin_ye100861 小时前
C# 零基础到精通教程 - WPF 专题三:高级控件与自定义控件
开发语言·c#·wpf
SoftLipaRZC1 小时前
C语言自定义类型:结构体完全指南
c语言·开发语言
方也_arkling1 小时前
【Java-Day19】集合3 List中常见的方法和5种遍历方式
java·开发语言
AI玫瑰助手1 小时前
Python函数:局部变量与全局变量的作用域
开发语言·python·信息可视化
我是一颗柠檬1 小时前
【Redis】主从复制Day9
java·数据库·redis·后端
8Qi81 小时前
LeetCode 208:实现 Trie(前缀树)—— Java 题解 ✅
java·算法·leetcode·二叉树·tire树