任务调度新境界:探秘ScheduledExecutorService的异步魔力

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 任务调度新境界:探秘ScheduledExecutorService的异步魔力

欢迎来到我的博客,代码的世界里,每一行都是一个故事


前言

在编程的世界里,我们经常需要让某些任务在未来的特定时间点执行。这就是ScheduledExecutorService登场的时刻,它是一个任务调度的专业管家,能够精确地掌握时间的舞步。让我们一同踏入这个时间的王国,探索其中的奇妙之处。

ScheduledExecutorService的基本概念

ScheduledExecutorService 是 Java 并发包提供的接口,用于支持任务的调度和执行。它是一个更强大、更灵活的定时任务调度工具,相较于传统的 Timer 类,ScheduledExecutorService 具有更多的功能和更好的性能。

基本概念:

  1. 定义: ScheduledExecutorService 接口是 ExecutorService 的子接口,用于在给定的时间延迟之后,或者周期性地执行任务。
  2. 基本原理: ScheduledExecutorService 使用线程池来管理和执行任务,可以异步地执行任务,支持延迟执行和周期性执行。

为何它是 Java 中任务调度的首选工具:

  1. 灵活性: ScheduledExecutorService 提供了更灵活的任务调度机制,可以支持延迟执行、周期性执行等多种调度方式。这使得它适用于各种不同的定时任务场景。
  2. 可控性: 通过使用线程池,ScheduledExecutorService 提供了对任务执行线程的管理和控制,能够更好地适应不同的并发需求。
  3. 异常处理:Timer 不同,ScheduledExecutorService 对于任务执行中的异常有更好的处理机制,不会因为一个任务的异常导致整个调度器终止。
  4. 相对线程安全: ScheduledExecutorService 在设计上相对于 Timer 更加线程安全,更适合在多线程环境中使用。
  5. 替代 Timer: 由于 ScheduledExecutorService 具有更多功能且更健壮,它通常被认为是 Timer 的替代品,特别是在需要更复杂调度需求和更好性能的情况下。
  6. ExecutorService 的扩展: 作为 ExecutorService 的子接口,ScheduledExecutorService 不仅可以执行定时任务,还能执行普通的异步任务,使得任务的管理更加一致和统一。

基本用法:

使用 ScheduledExecutorService 的基本流程如下:

  1. 创建 ScheduledExecutorService 实例:
ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
  1. 创建任务(实现 RunnableCallable 接口):
Runnable task = () -> {
    // 任务逻辑
    System.out.println("Task executed at: " + System.currentTimeMillis());
};
  1. 安排任务的执行:
  • 在延迟一定时间后执行任务:
scheduledExecutorService.schedule(task, 1, TimeUnit.SECONDS); // 1秒后执行
  • 周期性执行任务:
scheduledExecutorService.scheduleAtFixedRate(task, 0, 2, TimeUnit.SECONDS); // 每2秒执行一次
  1. 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();

总体而言,ScheduledExecutorService 提供了更灵活和强大的任务调度功能,是 Java 中任务调度的首选工具之一。

ScheduledExecutorService的创建与配置

ScheduledExecutorService 的创建和配置通常通过 Executors 工厂类完成。下面是一个基本的实例化和配置 ScheduledExecutorService 的例子:

import java.util.concurrent.*;
public class ScheduledExecutorServiceExample {
    public static void main(String[] args) {
        // 创建一个具有固定线程数的 ScheduledExecutorService
        ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(3);
        // 创建任务
        Runnable task = () -> {
            // 任务逻辑
            System.out.println("Task executed at: " + System.currentTimeMillis());
        };
        // 配置任务的执行方式
        ScheduledFuture<?> scheduledFuture = scheduledExecutorService.scheduleAtFixedRate(
                task,    // 任务
                0,       // 初始延迟
                2,       // 间隔时间
                TimeUnit.SECONDS  // 时间单位
        );
        // 关闭 ScheduledExecutorService
        scheduledExecutorService.shutdown();
    }
}

在上述例子中,我们通过 Executors.newScheduledThreadPool(3) 创建了一个固定线程数为 3 的 ScheduledExecutorService。接着,我们定义了一个简单的任务 task,并使用 scheduleAtFixedRate 方法配置了任务的执行方式。最后,我们通过 shutdown 方法关闭了 ScheduledExecutorService

配置项:

newScheduledThreadPool 方法允许你传递一个整数参数,用于指定线程池的大小。这个参数表示同时执行的线程数,也即池中的最大线程数。除了这个参数外,newScheduledThreadPool 方法还允许你传递一个 ThreadFactory 对象,用于创建线程。

对于更高级的配置,可以使用 ScheduledThreadPoolExecutor 的构造函数,允许你手动配置线程池的各种参数,如核心线程数、最大线程数、线程空闲时间等。

ScheduledExecutorService scheduledExecutorService = new ScheduledThreadPoolExecutor(
        corePoolSize,      // 核心线程数
        threadFactory,      // 线程工厂
        handler             // 拒绝策略
);

其中,corePoolSize 是核心线程数,threadFactory 是线程工厂,handler 是拒绝策略。这样的创建方式更为灵活,可以根据实际需求进行配置。

任务的添加与取消

ScheduledExecutorService 中,可以使用不同的方法来添加和取消定时任务。以下是添加和取消定时任务的基本方法:

添加定时任务:

  1. 使用 schedule 方法:
ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
Runnable task = () -> {
    // 任务逻辑
    System.out.println("Task executed at: " + System.currentTimeMillis());
};
// 在延迟一定时间后执行任务
ScheduledFuture<?> scheduledFuture = scheduledExecutorService.schedule(task, 1, TimeUnit.SECONDS);
// 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();
  1. 使用 scheduleAtFixedRatescheduleWithFixedDelay 方法:
ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
Runnable task = () -> {
    // 任务逻辑
    System.out.println("Task executed at: " + System.currentTimeMillis());
};
// 周期性执行任务,scheduleAtFixedRate 方法
// 或者使用 scheduleWithFixedDelay 方法
// 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();

取消定时任务:

  1. 使用 ScheduledFuture 对象取消任务:
ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
Runnable task = () -> {
    // 任务逻辑
    System.out.println("Task executed at: " + System.currentTimeMillis());
};
// 在延迟一定时间后执行任务
ScheduledFuture<?> scheduledFuture = scheduledExecutorService.schedule(task, 1, TimeUnit.SECONDS);
// 取消任务
boolean cancelled = scheduledFuture.cancel();
// 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();
  1. cancel 方法返回一个布尔值,表示任务是否被取消成功。如果任务已经开始执行或已经完成,取消操作将失败。
  2. 使用 shutdownNow 方法取消所有任务:
ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
Runnable task = () -> {
    // 任务逻辑
    System.out.println("Task executed at: " + System.currentTimeMillis());
};
// 在延迟一定时间后执行任务
ScheduledFuture<?> scheduledFuture = scheduledExecutorService.schedule(task, 1, TimeUnit.SECONDS);
// 取消所有任务
List<Runnable> cancelledTasks = scheduledExecutorService.shutdownNow();
// 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();
  1. shutdownNow 方法返回一个 List<Runnable>,包含所有被取消的任务。

注意事项:

  • 使用 cancel 方法时,需要注意任务是否已经开始执行或已经完成。取消操作只在任务尚未开始执行时才能成功。
  • 在使用 shutdownNow 方法取消所有任务时,可能会中断正在执行的任务。因此,需要确保任务的设计和实现能够处理中断。

不同类型的定时任务

ScheduledExecutorService 中,有多种方法可以配置不同类型的定时任务,包括定时执行任务、固定频率执行任务等。以下是不同类型的定时任务以及使用不同的方法配置的示例:

定时执行任务:

ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
Runnable task = () -> {
    // 任务逻辑
    System.out.println("Task executed at: " + System.currentTimeMillis());
};
// 在延迟一定时间后执行任务
ScheduledFuture<?> scheduledFuture = scheduledExecutorService.schedule(task, 1, TimeUnit.SECONDS);
// 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();

上述代码中,schedule 方法用于在延迟一定时间后执行任务,即定时执行任务。

固定频率执行任务:

ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
Runnable task = () -> {
    // 任务逻辑
    System.out.println("Task executed at: " + System.currentTimeMillis());
};
// 周期性执行任务,scheduleAtFixedRate 方法
ScheduledFuture<?> scheduledFuture = scheduledExecutorService.scheduleAtFixedRate(
        task,    // 任务
        0,       // 初始延迟
        2,       // 间隔时间
        TimeUnit.SECONDS  // 时间单位
);
// 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();

上述代码中,scheduleAtFixedRate 方法用于周期性地执行任务,可以指定初始延迟和执行间隔。

固定延迟执行任务:

ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
Runnable task = () -> {
    // 任务逻辑
    System.out.println("Task executed at: " + System.currentTimeMillis());
};
// 固定延迟执行任务,scheduleWithFixedDelay 方法
ScheduledFuture<?> scheduledFuture = scheduledExecutorService.scheduleWithFixedDelay(
        task,    // 任务
        0,       // 初始延迟
        2,       // 间隔时间
        TimeUnit.SECONDS  // 时间单位
);
// 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();

上述代码中,scheduleWithFixedDelay 方法用于固定延迟地执行任务,即任务执行完毕后等待指定的时间再执行下一次。

注意事项:

  • 定时任务的配置方法根据具体需求选择。schedule 适用于延迟一定时间后执行一次的任务,scheduleAtFixedRate 适用于周期性执行任务,而 scheduleWithFixedDelay 适用于固定延迟执行任务。
  • 在使用这些方法时,需要考虑任务的执行时间和任务之间的依赖关系,以确保任务能够按照预期执行。

异常处理与容错机制

ScheduledExecutorService 中,处理任务执行中的异常是关键的一部分,以确保定时任务的稳定性。以下是一些处理异常和容错机制的方法:

1. 异常处理:

在任务的 run 方法中进行异常处理是一种常见的做法,可以使用 try-catch 块捕获异常,并在异常发生时执行适当的处理逻辑。例如,记录日志、发送警报或执行备用逻辑。

Runnable task = () -> {
    try {
        // 任务逻辑
        // ...
    } catch (Exception e) {
        // 异常处理逻辑
        // 记录日志、发送警报等
        e.printStackTrace();
    }
};

2. 使用 UncaughtExceptionHandler:

ScheduledThreadPoolExecutor 类提供了 setUncaughtExceptionHandler 方法,可以设置一个全局的未捕获异常处理器。这个处理器将在任务抛出未捕获的异常时被调用。

Thread.UncaughtExceptionHandler exceptionHandler = (thread, throwable) -> {
    // 全局未捕获异常处理逻辑
    // 记录日志、发送警报等
    throwable.printStackTrace();
};
ScheduledExecutorService scheduledExecutorService = new ScheduledThreadPoolExecutor(1);
((ScheduledThreadPoolExecutor) scheduledExecutorService).setUncaughtExceptionHandler(exceptionHandler);
Runnable task = () -> {
    // 任务逻辑
    // ...
};

3. 封装任务逻辑:

将任务逻辑封装在一个方法中,并在方法内进行异常处理。这样可以使任务逻辑更加清晰,异常处理也更为集中。

Runnable task = () -> {
    try {
        // 封装的任务逻辑
        executeTask();
    } catch (Exception e) {
        // 异常处理逻辑
        // 记录日志、发送警报等
        e.printStackTrace();
    }
};
private void executeTask() {
    // 具体的任务逻辑
    // ...
}

4. 返回 Future 对象:

ScheduledExecutorServiceschedule 方法返回一个 ScheduledFuture 对象,可以使用这个对象检查任务的执行状态和获取任务的执行结果。通过检查 ScheduledFuture 对象,可以在任务执行失败时获取异常信息。

ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
Runnable task = () -> {
    // 任务逻辑
    // ...
};
ScheduledFuture<?> scheduledFuture = scheduledExecutorService.schedule(task, 1, TimeUnit.SECONDS);
try {
    // 获取任务执行结果,这里会抛出异常,可以在这里处理异常
    scheduledFuture.get();
} catch (Exception e) {
    // 异常处理逻辑
    // 记录日志、发送警报等
    e.printStackTrace();
}
// 关闭 ScheduledExecutorService
scheduledExecutorService.shutdown();

5. 合理的重试机制:

在异常发生时,可以考虑使用重试机制,即在一定次数内尝试重新执行任务。这可以通过在任务逻辑中使用循环来实现。

Runnable task = () -> {
    int maxAttempts = 3;
    for (int attempt = 1; attempt <= maxAttempts; attempt++) {
        try {
            // 任务逻辑
            // ...
            break; // 任务成功执行,跳出循环
        } catch (Exception e) {
            // 异常处理逻辑
            // 记录日志、发送警报等
            e.printStackTrace();
            if (attempt < maxAttempts) {
                // 等待一段时间后重试
                Thread.sleep(1000);
            } else {
                // 达到最大重试次数,放弃任务执行
                break;
            }
        }
    }
};

注意事项:

  • 在异常处理中,需要根据具体业务需求选择合适的处理方式,例如记录日志、发送警报、重试等。
  • 在定时任务中,为了确保任务执行的稳定性,合理的异常处理和容错机制是至关重要的。

ScheduledExecutorService的优势与劣势

优势:

  1. 灵活性: ScheduledExecutorService 提供了更灵活的任务调度机制,支持延迟执行、周期性执行等多种调度方式。这使得它适用于各种不同的定时任务场景。
  2. 可控性: 通过使用线程池,ScheduledExecutorService 提供了对任务执行线程的管理和控制,能够更好地适应不同的并发需求。
  3. 异常处理: 相对于 TimerScheduledExecutorService 对于任务执行中的异常有更好的处理机制,不会因为一个任务的异常导致整个调度器终止。
  4. 相对线程安全: ScheduledExecutorService 在设计上相对于 Timer 更加线程安全,更适合在多线程环境中使用。
  5. ExecutorService 的扩展: 作为 ExecutorService 的子接口,ScheduledExecutorService 不仅可以执行定时任务,还能执行普通的异步任务,使得任务的管理更加一致和统一。
  6. 更好的性能: 相较于 TimerScheduledExecutorService 的性能通常更好。它能够更好地处理任务的并发执行,提高系统的吞吐量。

局限性与风险:

  1. 不适用于复杂场景: 对于一些复杂的任务调度场景,例如需要更高级的调度策略、任务间的依赖关系等,ScheduledExecutorService 可能显得力不从心,因为其功能相对有限。
  2. 定时器线程生命周期管理: ScheduledExecutorService 的定时器线程在 shutdown 方法被调用后不会被及时终止,可能导致应用程序无法正常退出。需要谨慎管理定时器线程的生命周期。
  3. 不支持任务的取消和修改: 一旦定时任务被安排,就不能取消或修改其执行时间,只能取消整个定时器并重新创建。
  4. 任务执行时间长: 如果某个任务的执行时间过长,可能会影响后续任务的调度,因为任务是按照顺序执行的。

总体而言,ScheduledExecutorService 是一个更灵活、更可控且相对线程安全的定时任务调度工具,适用于大多数场景。然而,在一些复杂的调度需求下,可能需要考虑使用其他更为高级的调度工具或框架。

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
分布式计算 并行计算 数据库
Schedulerx2.0分布式计算原理&最佳实践
1. 前言 Schedulerx2.0的客户端提供分布式执行、多种任务类型、统一日志等框架,用户只要依赖schedulerx-worker这个jar包,通过schedulerx2.0提供的编程模型,简单几行代码就能实现一套高可靠可运维的分布式执行引擎。
23694 2
|
6月前
|
消息中间件 安全 Java
一起来探究@Schedule定时任务在分布式产生的问题
一起来探究@Schedule定时任务在分布式产生的问题
383 0
|
分布式计算 前端开发 数据可视化
你只会用 xxl-job?一款更强大、新一代分布式任务调度框架来了,太强大了!
你只会用 xxl-job?一款更强大、新一代分布式任务调度框架来了,太强大了!
888 0
你只会用 xxl-job?一款更强大、新一代分布式任务调度框架来了,太强大了!
|
JSON 缓存 JavaScript
提高系统吞吐量的一把利器:DeferredResult 到底有多强?
提高系统吞吐量的一把利器:DeferredResult 到底有多强?
|
Java 测试技术 调度
【优化技术专题】「温故而知新」基于Quartz系列的任务调度框架的动态化任务实现分析
【优化技术专题】「温故而知新」基于Quartz系列的任务调度框架的动态化任务实现分析
153 0
【优化技术专题】「温故而知新」基于Quartz系列的任务调度框架的动态化任务实现分析
|
运维 NoSQL 数据库连接
定时任务能力进击!Quartz框架的使用
定时任务能力进击!Quartz框架的使用
|
SQL 分布式计算 Java
Dolphinscheduler海豚调度器实现离线任务提交安装实录
学习一个东西,个人认为最好的方式是:官网+源码+实践。
2326 0
Dolphinscheduler海豚调度器实现离线任务提交安装实录
|
资源调度 分布式计算 运维
阿里巴巴任务调度SchedulerX支持一次性任务
阿里巴巴任务调度SchedulerX2.0支持一次性任务
1229 2
|
Java API 调度
一文搞懂Executor执行器和线程池的关系,整体介绍其任务执行/调度体系:ThreadPoolExecutor、ScheduledExecutorService(下)
一文搞懂Executor执行器和线程池的关系,整体介绍其任务执行/调度体系:ThreadPoolExecutor、ScheduledExecutorService(下)
一文搞懂Executor执行器和线程池的关系,整体介绍其任务执行/调度体系:ThreadPoolExecutor、ScheduledExecutorService(下)
|
Java 数据库连接 API
一文搞懂Executor执行器和线程池的关系,整体介绍其任务执行/调度体系:ThreadPoolExecutor、ScheduledExecutorService(上)
一文搞懂Executor执行器和线程池的关系,整体介绍其任务执行/调度体系:ThreadPoolExecutor、ScheduledExecutorService(上)
一文搞懂Executor执行器和线程池的关系,整体介绍其任务执行/调度体系:ThreadPoolExecutor、ScheduledExecutorService(上)