SpringBoot基于数据库实现定时任务过程解析
下面是关于“Spring Boot基于数据库实现定时任务过程解析”的完整攻略。
1. 背景
定时任务在业务逻辑中经常被使用,而且很多时候任务调度需要依赖于数据库中的数据。Spring Boot中提供了很方便的方式来实现定时任务的功能,而且也支持基于数据库的方式来实现任务调度,本文将详细讲解如何使用Spring Boot实现基于数据库的定时任务调度。
2. 实现过程
2.1 添加依赖
首先,我们需要在Spring Boot项目的pom.xml文件中添加相关依赖:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>org.quartz-scheduler</groupId>
<artifactId>quartz</artifactId>
<version>2.3.2</version>
</dependency>
<dependency>
<groupId>org.quartz-scheduler</groupId>
<artifactId>quartz-jobs</artifactId>
<version>2.3.2</version>
</dependency>
2.2 定义任务表
我们需要在数据库中定义一个任务表,用来存储任务的具体信息。可以使用如下的SQL语句创建该表:
CREATE TABLE `tb_task` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '任务编号',
`task_name` varchar(100) DEFAULT NULL COMMENT '任务名称',
`task_group` varchar(50) DEFAULT NULL COMMENT '任务所属分组',
`task_class` varchar(255) DEFAULT NULL COMMENT '任务类名',
`task_method` varchar(50) DEFAULT NULL COMMENT '任务方法名',
`task_params` varchar(255) DEFAULT NULL COMMENT '任务参数',
`task_expression` varchar(255) DEFAULT NULL COMMENT '任务表达式',
`task_description` varchar(255) DEFAULT NULL COMMENT '任务描述',
`create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
`update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
PRIMARY KEY (`id`),
KEY `idx_task_name` (`task_name`) USING BTREE,
KEY `idx_task_group` (`task_group`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='定时任务表';
2.3 定义实体类
我们需要定义一个实体类,用来封装任务表的字段信息,代码如下:
@Entity
@Table(name = "tb_task")
public class Task {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String taskName;
private String taskGroup;
private String taskClass;
private String taskMethod;
private String taskParams;
private String taskExpression;
private String taskDescription;
private Date createTime;
private Date updateTime;
// getter and setter
}
2.4 定义定时任务执行逻辑
我们需要定义一个任务执行逻辑,具体的执行逻辑在该类的execute方法中实现,代码如下:
@Component
public class SimpleTask implements Job {
@Override
public void execute(JobExecutionContext context) throws JobExecutionException {
// 具体的任务执行逻辑
// 可以从context中获取任务的参数等信息,具体的实现方式可以参考quartz的文档
}
}
2.5 定义任务调度器
我们需要定义一个任务调度器,用来实现从任务表中读取任务信息,并将任务加入到任务调度器中,代码如下:
@Component
public class TaskScheduler {
private final Logger logger = LoggerFactory.getLogger(TaskScheduler.class);
@Autowired
private Scheduler scheduler;
@Autowired
private TaskRepository taskRepository;
@Autowired
private ApplicationContext applicationContext;
public void schedule() {
List<Task> tasks = taskRepository.findAll();
if (tasks != null && tasks.size() > 0) {
for (Task task : tasks) {
String taskClass = task.getTaskClass();
String taskMethod = task.getTaskMethod();
String taskParams = task.getTaskParams();
String taskExpression = task.getTaskExpression();
try {
// 创建JobDetail
JobDetail jobDetail = JobBuilder.newJob()
.ofType((Class<? extends Job>) Class.forName(taskClass))
.withIdentity(task.getTaskName(), task.getTaskGroup())
.build();
// 设置JobDataMap
JobDataMap jobDataMap = jobDetail.getJobDataMap();
jobDataMap.put("applicationContext", applicationContext);
jobDataMap.put("taskParams", taskParams);
// 创建CronTrigger
CronTrigger trigger = TriggerBuilder.newTrigger()
.withIdentity(task.getTaskName(), task.getTaskGroup())
.withSchedule(CronScheduleBuilder.cronSchedule(taskExpression))
.build();
scheduler.scheduleJob(jobDetail, trigger);
logger.info("任务已添加:{}", task.getTaskName());
} catch (Exception e) {
logger.error("添加任务失败:{}", task.getTaskName(), e);
}
}
try {
scheduler.start();
logger.info("任务调度器已启动");
} catch (SchedulerException e) {
logger.error("启动任务调度器失败", e);
}
}
}
}
2.6 启动任务调度器
最后,在Spring Boot项目的入口处添加如下代码,来启动任务调度器:
@SpringBootApplication
public class Application implements CommandLineRunner {
@Autowired
private TaskScheduler taskScheduler;
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
@Override
public void run(String... args) throws Exception {
taskScheduler.schedule();
}
}
3. 示例说明
3.1 示例1:每天0点执行一次任务
假设我们需要每天的0点执行一次任务,我们可以先向任务表中添加一条数据,具体数据如下:
task_name = "daily_task"
task_group = "group1"
task_class = "com.example.task.SimpleTask"
task_method = "execute"
task_params = "param1,param2" // 参数1和参数2使用逗号分隔
task_expression = "0 0 0 * * ?" // 任务将在每天的0点执行
task_description = "每天0点执行一次任务"
当我们启动Spring Boot应用时,任务调度器会从任务表中读取该任务信息,并将任务加入到任务调度器中。当系统时间到达每天的0点时,任务调度器会自动执行对应的任务。
3.2 示例2:每间隔10秒执行一次任务
假设我们需要每间隔10秒执行一次任务,我们可以先向任务表中添加一条数据,具体数据如下:
task_name = "every_10s_task"
task_group = "group2"
task_class = "com.example.task.SimpleTask"
task_method = "execute"
task_params = "param1,param2" // 参数1和参数2使用逗号分隔
task_expression = "0/10 * * * * ?" // 任务将每间隔10秒执行一次
task_description = "每间隔10秒执行一次任务"
当我们启动Spring Boot应用时,任务调度器会从任务表中读取该任务信息,并将任务加入到任务调度器中。当系统时间到达每10秒的整数倍时,任务调度器会自动执行对应的任务。
4. 总结
通过本文的介绍,我们可以发现基于数据库实现定时任务的方式非常灵活,可以根据具体的业务需求来灵活配置任务的参数和执行规则。同时,Quartz作为一款优秀的任务调度框架,在Spring Boot中支持良好,可以帮助我们更加方便地实现定时任务的功能。