当前位置: 首页 > news >正文

Spring Boot中使用ThreadPoolTaskScheduler实现轻量级多线程定时任务

引言

在Java开发中,Spring Boot提供了多种方式来执行定时任务,如@Scheduled注解和TaskScheduler。当需要执行多线程定时任务时,ThreadPoolTaskScheduler是一个轻量级的解决方案。本文将通过一个具体的业务场景,介绍如何使用ThreadPoolTaskScheduler来实现多线程定时任务。

业务场景

假设我们需要开发一个电商平台,该平台需要定期执行以下任务:

  1. 库存检查:每天检查库存量,如果低于阈值,则自动触发补货。

  2. 订单审核:每小时审核新订单,自动确认有效订单。

这些任务需要并行执行,以提高效率。

实现步骤

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. 创建定时任务

接下来,我们创建具体的定时任务。我们将实现两个服务:InventoryCheckServiceOrderAuditService

 

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&#xff…...

每日一题学习笔记

给你两个字符串: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是一款针对漏洞奖励计划的子域名安全枚举工具&#xff0c;可用于在漏洞搜寻和侦察过程中进行被动子域名枚举。它旨在通过高效枚举子域名和各种免费被动资源来帮助研究人员和网络安全专业人员发现潜在的安全漏洞。 Subdominator 与各种免费和付费…...

[leetcode]516_最长回文子序列

给你一个字符串 s &#xff0c;找出其中最长的回文子序列&#xff0c;并返回该序列的长度。 子序列定义为&#xff1a;不改变剩余字符顺序的情况下&#xff0c;删除某些字符或者不删除任何字符形成的一个序列。示例 1&#xff1a; 输入&#xff1a;s "bbbab" 输出&a…...

电子相册|智能化电子相册|基于java的电子相册管理系统设计与实现(源码+数据库+文档)

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

linux项目_c语言:Makefile编写、动态库生成、添加动态库路径

一直想搞懂Linux中Makefile是怎么管理项目的&#xff0c;知识积累到一定程度后&#xff0c;我就做了一个自己的缩小项目去把剩下的细节搞清楚 代码&#xff1a; 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) # 输出&#xff1a;{name: Alice, age: 25} 1.2 使用 dict() 函数 # 使用键值对列表创建字典 person dict(name"…...

async await 介绍 从0手动实现async await

1 async await介绍 async 和 await 是用于处理异步编程的语法糖&#xff0c;它们简化了异步操作的编写&#xff0c;使其看起来像同步代码。通过 async 标记一个函数为异步函数&#xff0c;返回的是一个 Promise 对象&#xff0c;而 await 用来暂停执行&#xff0c;直到 Promise…...

UDP校验和计算及网络中的校验和机制

UDP (User Datagram Protocol) 是一种无连接的传输层协议&#xff0c;它不像 TCP 那样提供可靠的传输保证。虽然 UDP 不保证数据可靠性&#xff0c;但它仍然提供了一个可选的校验和机制来检测数据在传输过程中出现的错误。 理解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 官方地址 视频/图像标注工具&#xff0c;很适合用于目标追踪任务 DarkLabel可以在视频和图像中标注物体的边界框&#xff0c;并附上 ID 和name。还可以用于裁剪视频、从视频中采样训练图像以及对图像区域进行马赛克处理…...

uniapp设置从右上角到左下角的三种渐变颜色

推荐学习文档 golang应用级os框架&#xff0c;欢迎stargolang应用级os框架使用案例&#xff0c;欢迎star案例&#xff1a;基于golang开发的一款超有个性的旅游计划app经历golang实战大纲golang优秀开发常用开源库汇总想学习更多golang知识&#xff0c;这里有免费的golang学习笔…...

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…...

手游刚开服就被攻击怎么办?如何防御DDoS?

开服初期是手游最脆弱的阶段&#xff0c;极易成为DDoS攻击的目标。一旦遭遇攻击&#xff0c;可能导致服务器瘫痪、玩家流失&#xff0c;甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案&#xff0c;帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...

应用升级/灾备测试时使用guarantee 闪回点迅速回退

1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间&#xff0c; 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点&#xff0c;不需要开启数据库闪回。…...

蓝桥杯 2024 15届国赛 A组 儿童节快乐

P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡&#xff0c;轻快的音乐在耳边持续回荡&#xff0c;小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下&#xff0c;六一来了。 今天是六一儿童节&#xff0c;小蓝老师为了让大家在节…...

学校招生小程序源码介绍

基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码&#xff0c;专为学校招生场景量身打造&#xff0c;功能实用且操作便捷。 从技术架构来看&#xff0c;ThinkPHP提供稳定可靠的后台服务&#xff0c;FastAdmin加速开发流程&#xff0c;UniApp则保障小程序在多端有良好的兼…...

【ROS】Nav2源码之nav2_behavior_tree-行为树节点列表

1、行为树节点分类 在 Nav2(Navigation2)的行为树框架中,行为树节点插件按照功能分为 Action(动作节点)、Condition(条件节点)、Control(控制节点) 和 Decorator(装饰节点) 四类。 1.1 动作节点 Action 执行具体的机器人操作或任务,直接与硬件、传感器或外部系统…...

Redis数据倾斜问题解决

Redis 数据倾斜问题解析与解决方案 什么是 Redis 数据倾斜 Redis 数据倾斜指的是在 Redis 集群中&#xff0c;部分节点存储的数据量或访问量远高于其他节点&#xff0c;导致这些节点负载过高&#xff0c;影响整体性能。 数据倾斜的主要表现 部分节点内存使用率远高于其他节…...

Android第十三次面试总结(四大 组件基础)

Activity生命周期和四大启动模式详解 一、Activity 生命周期 Activity 的生命周期由一系列回调方法组成&#xff0c;用于管理其创建、可见性、焦点和销毁过程。以下是核心方法及其调用时机&#xff1a; ​onCreate()​​ ​调用时机​&#xff1a;Activity 首次创建时调用。​…...

【电力电子】基于STM32F103C8T6单片机双极性SPWM逆变(硬件篇)

本项目是基于 STM32F103C8T6 微控制器的 SPWM(正弦脉宽调制)电源模块,能够生成可调频率和幅值的正弦波交流电源输出。该项目适用于逆变器、UPS电源、变频器等应用场景。 供电电源 输入电压采集 上图为本设计的电源电路,图中 D1 为二极管, 其目的是防止正负极电源反接, …...

MySQL 部分重点知识篇

一、数据库对象 1. 主键 定义 &#xff1a;主键是用于唯一标识表中每一行记录的字段或字段组合。它具有唯一性和非空性特点。 作用 &#xff1a;确保数据的完整性&#xff0c;便于数据的查询和管理。 示例 &#xff1a;在学生信息表中&#xff0c;学号可以作为主键&#xff…...