Spring Boot中使用ThreadPoolTaskScheduler实现轻量级多线程定时任务
引言
在Java开发中,Spring Boot提供了多种方式来执行定时任务,如@Scheduled
注解和TaskScheduler
。当需要执行多线程定时任务时,ThreadPoolTaskScheduler
是一个轻量级的解决方案。本文将通过一个具体的业务场景,介绍如何使用ThreadPoolTaskScheduler
来实现多线程定时任务。
业务场景
假设我们需要开发一个电商平台,该平台需要定期执行以下任务:
-
库存检查:每天检查库存量,如果低于阈值,则自动触发补货。
-
订单审核:每小时审核新订单,自动确认有效订单。
这些任务需要并行执行,以提高效率。
实现步骤
1. 配置ThreadPoolTaskScheduler
首先,我们需要配置ThreadPoolTaskScheduler
。在Spring Boot应用中,我们可以创建一个配置类来实现这一点。
java
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;@Configuration
@EnableScheduling
public class SchedulerConfig implements SchedulingConfigurer {@Overridepublic void configureTasks(ThreadPoolTaskScheduler taskScheduler) {taskScheduler.setPoolSize(10); // 设置线程池大小taskScheduler.setThreadNamePrefix("task-scheduler-");taskScheduler.initialize();}
}
2. 创建定时任务
接下来,我们创建具体的定时任务。我们将实现两个服务:InventoryCheckService
和OrderAuditService
。
java
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;@Service
public class InventoryCheckService {@Scheduled(cron = "0 0 1 * * ?") // 每天凌晨1点执行public void checkInventory() {System.out.println("执行库存检查任务");// 库存检查逻辑}
}
java
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;@Service
public class OrderAuditService {@Scheduled(cron = "0 0 * * * ?") // 每小时执行public void auditOrders() {System.out.println("执行订单审核任务");// 订单审核逻辑}
}
3. 多线程执行定时任务
为了使定时任务并行执行,我们可以在Scheduled
注解中设置taskScheduler
属性。
java
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class InventoryCheckService {@Autowiredprivate TaskScheduler taskScheduler;@Scheduled(cron = "0 0 1 * * ?", taskScheduler = "threadPoolTaskScheduler")public void checkInventory() {taskScheduler.execute(() -> {System.out.println("执行库存检查任务");// 库存检查逻辑});}
}
java
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class OrderAuditService {@Autowiredprivate TaskScheduler taskScheduler;@Scheduled(cron = "0 0 * * * ?", taskScheduler = "threadPoolTaskScheduler")public void auditOrders() {taskScheduler.execute(() -> {System.out.println("执行订单审核任务");// 订单审核逻辑});}
}
4. 启动定时任务
确保您的Spring Boot应用已经启用了定时任务的支持,通过在启动类上添加@EnableScheduling
注解。
java
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableScheduling;@SpringBootApplication
@EnableScheduling
public class Application {public static void main(String[] args) {SpringApplication.run(Application.class, args);}
}
异常处理
在处理定时任务中的异常情况时,确保系统稳定性的关键在于正确地捕获和处理这些异常。以下是一些最佳实践:
1. 使用try-catch
块
确保每个定时任务都被try-catch
块包裹,以便在任务执行过程中捕获任何未预料到的异常。
java
@Scheduled(fixedRate = 5000)
public void performTask() {try {// 任务逻辑} catch (Exception e) {// 异常处理逻辑logger.error("Error executing scheduled task", e);}
}
2. 记录异常
当异常发生时,记录异常信息对于调试和监控系统状态非常重要。
java
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;public class MyScheduledTask {private static final Logger logger = LoggerFactory.getLogger(MyScheduledTask.class);@Scheduled(cron = "0 * * * * *")public void scheduledTask() {try {// 任务逻辑} catch (Exception ex) {logger.error("Scheduled task failed", ex);}}
}
3. 重试机制
对于某些类型的错误,可以实施重试策略。例如,如果一个任务因为网络问题失败,可能在稍后重试时会成功。
java
@Scheduled(cron = "0 * * * * *")
public void scheduledTaskWithRetry() {try {// 任务逻辑} catch (TransientException te) {// 可以重试的异常scheduledTaskWithRetry(); // 重试逻辑} catch (Exception e) {// 非重试异常logger.error("Scheduled task failed", e);}
}
4. 错误隔离
如果可能,隔离可能导致异常的任务,以防止它们影响其他任务。
java
@Scheduled(cron = "0 * * * * *")
public void isolatedTask() {Thread taskThread = new Thread(() -> {try {// 任务逻辑} catch (Exception e) {logger.error("Isolated scheduled task failed", e);}});taskThread.setUncaughtExceptionHandler((thread, throwable) -> logger.error("Thread failed", throwable));taskThread.start();
}
5. 任务持久化
对于关键任务,可以考虑将任务持久化到数据库中,以便在系统崩溃后能够重新启动任务。
java
@Scheduled(cron = "0 * * * * *")
public void persistentTask() {try {// 任务逻辑} catch (Exception e) {logger.error("Persistent scheduled task failed", e);// 将任务状态标记为失败,并持久化}
}
6. 使用@Async
进行异步处理
使用@Async
注解将任务提交到Executor
,这样可以异步处理任务,并避免阻塞主线程。
java
@Async
@Scheduled(cron = "0 * * * * *")
public void asyncScheduledTask() {try {// 任务逻辑} catch (Exception e) {logger.error("Async scheduled task failed", e);}
}
7. 超时机制
为任务设置合理的超时时间,防止某些任务长时间占用资源。
java
@Scheduled(cron = "0 * * * * *")
public void timedTask() {try {// 任务逻辑} catch (TimeoutException te) {logger.warn("Scheduled task timed out", te);}
}
8. 监控和报警
实现监控机制,当任务失败时触发报警,以便及时响应。
java
@Scheduled(cron = "0 * * * * *")
public void monitoredTask() {try {// 任务逻辑} catch (Exception e) {logger.error("Monitored scheduled task failed", e);// 发送报警}
}
9. 回滚机制
对于涉及事务的任务,确保在异常时能够进行回滚。
java
@Transactional
@Scheduled(cron = "0 * * * * *")
public void transactionalTask() {try {// 任务逻辑} catch (Exception e) {logger.error("Transactional scheduled task failed", e);// 事务回滚}
}
10. 错误恢复策略
制定错误恢复策略,比如重试、重置状态、通知管理员等。
java
@Scheduled(cron = "0 * * * * *")
public void recoveryTask() {try {// 任务逻辑} catch (Exception e) {recoverFromError(); // 自定义恢复逻辑}
}
通过实施上述策略,可以显著提高定时任务的稳定性和可靠性,确保系统在面对异常情况时能够保持稳定运行。
结论
通过使用ThreadPoolTaskScheduler
,我们可以轻松地在Spring Boot应用中实现轻量级的多线程定时任务。这种方法不仅提高了任务执行的效率,而且使代码结构更加清晰。
相关文章:
Spring Boot中使用ThreadPoolTaskScheduler实现轻量级多线程定时任务
引言 在Java开发中,Spring Boot提供了多种方式来执行定时任务,如Scheduled注解和TaskScheduler。当需要执行多线程定时任务时,ThreadPoolTaskScheduler是一个轻量级的解决方案。本文将通过一个具体的业务场景,介绍如何使用Thread…...

完全二叉树的节点个数 C++ 简单问题
完全二叉树 的定义如下:在完全二叉树中,除了最底层节点可能没填满外,其余每层节点数都达到最大值,并且最下面一层的节点都集中在该层最左边的若干位置。若最底层为第 h 层,则该层包含 1~ 2h 个节点。 示例 1ÿ…...
每日一题学习笔记
给你两个字符串:ransomNote 和 magazine ,判断 ransomNote 能不能由 magazine 里面的字符构成。 如果可以,返回 true ;否则返回 false 。 magazine 中的每个字符只能在 ransomNote 中使用一次。 示例 1: 输入&#…...

从事人工智能学习Python还是学习C++?
人工智能(Artificial Intelligence,简称AI)是当今科技领域最热门的研究方向之一。AI 涉及多个学科和技术,特别是机器学习、神经网络、深度学习等技术的应用。在AI的开发过程中,编程语言的选择对于开发效率和项目实现至…...
博客摘录「 CNN中的感受野和有效感受野会对模型产生怎样的影响?」2024年9月29日
,中心像素受影响较大,离中心越远梯度信号越弱。梯度信号的衰减是指数级的,这意味着应用于感受野的大多数像素的梯度将是可忽略的(如果有的话)。 有效感受野的定义...

AURIX单片机示例:开发入门与点亮LED
文章目录 目的模板工程Blinky_LED示例链接总结 目的 这个例程比较简单,主要通过这个例程来介绍 AURIX™ Development Studio(ADS) 和 iLLD 库来开发 AURIX 系列单片机一些入门的内容。一些更为基础的资料等内容可以参考下面文章: 《英飞凌 AURIX TriCo…...
MySQL字符串函数与操作
在编程领域中,字符串操作是数据处理中至关重要的一部分。无论是文本分析、日志处理,还是格式化输出,字符串的操作技能都能极大提高工作效率。在 Python 中,字符串相关的函数和方法为开发者提供了强大的工具,帮助完成各种任务。了解如何灵活运用这些工具,能够有效提升编程…...

HTML+CSS 水滴登录页
文章目录 一、效果演示二、Code1.HTML2.CSS 三、实现思路拆分 一、效果演示 实现了一个水滴登录页的效果。页面包含一个水滴形状的登录框和两个按钮,登录框包括用户名、密码和登录按钮,按钮分别为忘记密码和注册。整个页面的设计非常有创意,采…...

基于Next.js和TailwindCss的TailwindCss
最近在研究 Next.js 和 TailwindCss ,这两天没事的时候就搞了一个 c。 目前工具部署在 Vercel ,欢迎各位体验(能提出意见更好嘿嘿) 体验地址: https://icon.999872.xyz/ 图片预览 👇...
若依开源系统多数据源整合clickhouse数据库详细步骤
1.添加依赖【pom.xml文件】 <!-- clickhouse数据源依赖--><dependency><groupId>ru.yandex.clickhouse</groupId>...

Subdominator:一款针对漏洞奖励计划的子域名安全枚举工具
关于Subdominator Subdominator是一款针对漏洞奖励计划的子域名安全枚举工具,可用于在漏洞搜寻和侦察过程中进行被动子域名枚举。它旨在通过高效枚举子域名和各种免费被动资源来帮助研究人员和网络安全专业人员发现潜在的安全漏洞。 Subdominator 与各种免费和付费…...
[leetcode]516_最长回文子序列
给你一个字符串 s ,找出其中最长的回文子序列,并返回该序列的长度。 子序列定义为:不改变剩余字符顺序的情况下,删除某些字符或者不删除任何字符形成的一个序列。示例 1: 输入:s "bbbab" 输出&a…...

电子相册|智能化电子相册|基于java的电子相册管理系统设计与实现(源码+数据库+文档)
电子相册管理系统 目录 基于java的电子相册管理系统设计与实现 一、前言 二、系统功能设计 三、系统实现 四、数据库设计 1、实体ER图 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取: 博主介绍:✌️大厂码农|毕设布道师&…...

linux项目_c语言:Makefile编写、动态库生成、添加动态库路径
一直想搞懂Linux中Makefile是怎么管理项目的,知识积累到一定程度后,我就做了一个自己的缩小项目去把剩下的细节搞清楚 代码: Service.c: #include <stdio.h> #include "lib_sevr.h" int main(){printf("输入a, b的值…...
Python学习(1):字典、DataFrame的创建方法
1. 字典的创建方法 1.1 直接创建 # 创建一个包含姓名和年龄的字典 person {"name": "Alice", "age": 25}print(person) # 输出:{name: Alice, age: 25} 1.2 使用 dict() 函数 # 使用键值对列表创建字典 person dict(name"…...
async await 介绍 从0手动实现async await
1 async await介绍 async 和 await 是用于处理异步编程的语法糖,它们简化了异步操作的编写,使其看起来像同步代码。通过 async 标记一个函数为异步函数,返回的是一个 Promise 对象,而 await 用来暂停执行,直到 Promise…...

UDP校验和计算及网络中的校验和机制
UDP (User Datagram Protocol) 是一种无连接的传输层协议,它不像 TCP 那样提供可靠的传输保证。虽然 UDP 不保证数据可靠性,但它仍然提供了一个可选的校验和机制来检测数据在传输过程中出现的错误。 理解UDP校验和的计算过程和其在网络中的作用至关重要。…...

如何使用C语言接入Doris数据库
如何使用C语言接入Doris数据库 一、环境准备1. 安装MySQL C API2. Doris数据库环境二、编写C语言接入代码1. 包含必要的头文件2. 编写连接和查询函数3. 编译和运行程序三、注意事项1. 安全性2. 错误处理3. 性能优化4. 兼容性5. 调试和日志记录四、结论Doris(之前称为Palo或Apa…...

DarkLabel 2.4 目标追标注工具介绍
DarkLabel介绍 https://github.com/darkpgmr/DarkLabel 官方地址 视频/图像标注工具,很适合用于目标追踪任务 DarkLabel可以在视频和图像中标注物体的边界框,并附上 ID 和name。还可以用于裁剪视频、从视频中采样训练图像以及对图像区域进行马赛克处理…...
uniapp设置从右上角到左下角的三种渐变颜色
推荐学习文档 golang应用级os框架,欢迎stargolang应用级os框架使用案例,欢迎star案例:基于golang开发的一款超有个性的旅游计划app经历golang实战大纲golang优秀开发常用开源库汇总想学习更多golang知识,这里有免费的golang学习笔…...

eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...

深入浅出Asp.Net Core MVC应用开发系列-AspNetCore中的日志记录
ASP.NET Core 是一个跨平台的开源框架,用于在 Windows、macOS 或 Linux 上生成基于云的新式 Web 应用。 ASP.NET Core 中的日志记录 .NET 通过 ILogger API 支持高性能结构化日志记录,以帮助监视应用程序行为和诊断问题。 可以通过配置不同的记录提供程…...
云计算——弹性云计算器(ECS)
弹性云服务器:ECS 概述 云计算重构了ICT系统,云计算平台厂商推出使得厂家能够主要关注应用管理而非平台管理的云平台,包含如下主要概念。 ECS(Elastic Cloud Server):即弹性云服务器,是云计算…...
Java 语言特性(面试系列1)
一、面向对象编程 1. 封装(Encapsulation) 定义:将数据(属性)和操作数据的方法绑定在一起,通过访问控制符(private、protected、public)隐藏内部实现细节。示例: public …...

聊聊 Pulsar:Producer 源码解析
一、前言 Apache Pulsar 是一个企业级的开源分布式消息传递平台,以其高性能、可扩展性和存储计算分离架构在消息队列和流处理领域独树一帜。在 Pulsar 的核心架构中,Producer(生产者) 是连接客户端应用与消息队列的第一步。生产者…...

Psychopy音频的使用
Psychopy音频的使用 本文主要解决以下问题: 指定音频引擎与设备;播放音频文件 本文所使用的环境: Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...
快刀集(1): 一刀斩断视频片头广告
一刀流:用一个简单脚本,秒杀视频片头广告,还你清爽观影体验。 1. 引子 作为一个爱生活、爱学习、爱收藏高清资源的老码农,平时写代码之余看看电影、补补片,是再正常不过的事。 电影嘛,要沉浸,…...

算法打卡第18天
从中序与后序遍历序列构造二叉树 (力扣106题) 给定两个整数数组 inorder 和 postorder ,其中 inorder 是二叉树的中序遍历, postorder 是同一棵树的后序遍历,请你构造并返回这颗 二叉树 。 示例 1: 输入:inorder [9,3,15,20,7…...

ui框架-文件列表展示
ui框架-文件列表展示 介绍 UI框架的文件列表展示组件,可以展示文件夹,支持列表展示和图标展示模式。组件提供了丰富的功能和可配置选项,适用于文件管理、文件上传等场景。 功能特性 支持列表模式和网格模式的切换展示支持文件和文件夹的层…...
Python常用模块:time、os、shutil与flask初探
一、Flask初探 & PyCharm终端配置 目的: 快速搭建小型Web服务器以提供数据。 工具: 第三方Web框架 Flask (需 pip install flask 安装)。 安装 Flask: 建议: 使用 PyCharm 内置的 Terminal (模拟命令行) 进行安装,避免频繁切换。 PyCharm Terminal 配置建议: 打开 Py…...