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中支持良好,可以帮助我们更加方便地实现定时任务的功能。

相关文章