spring - mvc - @Scheduled

@Scheduled

1.启用调度支持

为了在Spring中启用对调度任务和@Scheduled注释的支持,我们可以使用Java启用样式注释:

@Configuration
@EnableScheduling
public class SpringConfig {
    ...
}

相反,我们可以在 XML 中做同样的事情:

<task:annotation-driven>

2.按固定延迟安排任务

让我们首先将任务配置为在固定延迟后运行:

@Scheduled(fixedDelay = 1000)
public void scheduleFixedDelayTask() {
    System.out.println(
      "Fixed delay task - " + System.currentTimeMillis() / 1000);
}

在这种情况下,上一次执行结束和下一次执行开始之间的持续时间是固定的。该任务总是等待,直到前一个任务完成。
当必须在再次运行之前完成先前的执行时,应使用此选项。

3.以固定速率安排任务

现在让我们以固定的时间间隔执行任务:

@Scheduled(fixedRate = 1000)
public void scheduleFixedRateTask() {
    System.out.println(
      "Fixed rate task - " + System.currentTimeMillis() / 1000);
}

当任务的每次执行都是独立的时,应使用此选项。
请注意,默认情况下计划任务不会并行运行。因此,即使我们使用fixedRate,在前一个任务完成之前也不会调用下一个任务。
如果我们想在计划任务中支持并行行为,我们需要添加@Async注解:

@EnableAsync
public class ScheduledFixedRateExample {
    @Async
    @Scheduled(fixedRate = 1000)
    public void scheduleFixedRateTaskAsync() throws InterruptedException {
        System.out.println(
          "Fixed rate task async - " + System.currentTimeMillis() / 1000);
        Thread.sleep(2000);
    }

}

现在,即使前一个任务尚未完成,该异步任务也会每秒调用一次。

4.固定速率与固定延迟

我们可以使用 Spring 的@Scheduled注解来运行计划任务,但是根据属性fixedDelay 和 fixedRate, 执行的性质会发生变化。
fixedDelay属性确保任务执行的完成时间和下一次任务执行的开始时间之间有 n 毫秒的延迟。
当我们需要确保始终只有一个任务实例运行时,此属性特别有用。对于依赖工作来说,这是相当有帮助的。
fixedRate属性每 n毫秒运行一次计划任务。它不检查该任务的任何先前执行情况。
当任务的所有执行都是独立的时,这非常有用。如果我们不希望超出内存和线程池的大小,fixedRate 应该会很方便。
尽管如此,如果传入的任务没有快速完成,它们最终可能会出现“内存不足异常”。

5.安排一个带有初始延迟的任务

接下来,让我们安排一个有延迟(以毫秒为单位)的任务:

@Scheduled(fixedDelay = 1000, initialDelay = 1000)
public void scheduleFixedRateWithInitialDelayTask() {
 
    long now = System.currentTimeMillis() / 1000;
    System.out.println(
      "Fixed rate task with one second initial delay - " + now);
}

请注意在此示例中我们如何同时使用fixedDelay和initialDelay。任务会在initialDelay值之后第一次执行,并且会按照fixedDelay继续执行。
当任务有需要完成的设置时,此选项很方便

6.使用 Cron 表达式安排任务

有时延迟和速率还不够,我们需要 cron 表达式的灵活性来控制任务的时间表:

@Scheduled(cron = "0 15 10 15 * ?")
public void scheduleTaskUsingCronExpression() {
 
    long now = System.currentTimeMillis() / 1000;
    System.out.println(
      "schedule tasks using cron jobs - " + now);
}

请注意,在此示例中,我们计划在每月 15 日上午 10:15 执行一个任务。
默认情况下,Spring 将使用服务器的本地时区作为 cron 表达式。但是,我们可以使用 zone属性来更改该时区:

@Scheduled(cron = "0 15 10 15 * ?", zone = "Europe/Paris")

通过此配置,Spring 将安排带注释的方法在巴黎时间每月 15 日上午 10:15 运行。

7.参数化时间表

对这些时间表进行硬编码很简单,但我们通常需要能够控制时间表,而无需重新编译和重新部署整个应用程序。
我们将使用 Spring 表达式来外部化任务的配置,并将它们存储在属性文件中。

// 固定延迟任务
@Scheduled(fixedDelayString = "${fixedDelay.in.milliseconds}"):
// 固定速率任务:
@Scheduled(fixedRateString = "${fixedRate.in.milliseconds}")
// 基于cron表达式的任务:
@Scheduled(cron = "${cron.expression}")

8.使用 XML 配置计划任务

Spring还提供了配置计划任务的XML方式。以下是用于设置这些的 XML 配置:

<!-- Configure the scheduler -->
<task:scheduler id="myScheduler" pool-size="10" />

<!-- Configure parameters -->
<task:scheduled-tasks scheduler="myScheduler">
    <task:scheduled ref="beanA" method="methodA" 
      fixed-delay="5000" initial-delay="1000" />
    <task:scheduled ref="beanB" method="methodB" 
      fixed-rate="5000" />
    <task:scheduled ref="beanC" method="methodC" 
      cron="*/5 * * * * MON-FRI" />
</task:scheduled-tasks>

9.在运行时动态设置延迟或速率

通常, @Scheduled注解的所有属性仅在 Spring 上下文启动时解析和初始化一次。
因此,当我们在 Spring 中使用@Scheduled注解时,不可能在运行时更改fixedDelay或fixedRate值。
不过,有一个解决方法。使用Spring的SchedulingConfigurer提供了一种更加可定制的方式,使我们有机会动态设置延迟或速率。
让我们创建一个 Spring 配置DynamicSchedulingConfig并实现SchedulingConfigurer接口:

@Configuration
@EnableScheduling
public class DynamicSchedulingConfig implements SchedulingConfigurer {

    @Autowired
    private TickService tickService;

    @Bean
    public Executor taskExecutor() {
        return Executors.newSingleThreadScheduledExecutor();
    }

    @Override
    public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
        taskRegistrar.setScheduler(taskExecutor());
        taskRegistrar.addTriggerTask(
          new Runnable() {
              @Override
              public void run() {
                  tickService.tick();
              }
          },
          new Trigger() {
              @Override
              public Date nextExecutionTime(TriggerContext context) {
                  Optional<Date> lastCompletionTime =
                    Optional.ofNullable(context.lastCompletionTime());
                  Instant nextExecutionTime =
                    lastCompletionTime.orElseGet(Date::new).toInstant()
                      .plusMillis(tickService.getDelay());
                  return Date.from(nextExecutionTime);
              }
          }
        );
    }
}

正如我们注意到的,借助ScheduledTaskRegistrar#addTriggerTask方法,我们可以添加一个Runnable任务和一个Trigger实现,以在每次执行结束后重新计算nextExecutionTime 。
此外,我们使用@EnableScheduling注释DynamicSchedulingConfig以使调度工作。
因此,我们安排TickService#tick方法在每次延迟后运行它,该延迟是在运行时由getDelay方法动态确定的。

10.并行运行任务

默认情况下,Spring 使用本地单线程调度程序来运行任务。因此,即使我们有多个@Scheduled方法,它们每个都需要等待线程完成执行上一个任务。
如果我们的任务真正独立,那么并行运行它们会更方便。为此,我们需要提供一个更适合我们需求的TaskScheduler :

@Bean
public TaskScheduler  taskScheduler() {
    ThreadPoolTaskScheduler threadPoolTaskScheduler = new ThreadPoolTaskScheduler();
    threadPoolTaskScheduler.setPoolSize(5);
    threadPoolTaskScheduler.setThreadNamePrefix("ThreadPoolTaskScheduler");
    return threadPoolTaskScheduler;
}

在上面的示例中,我们将TaskScheduler配置为池大小为 5,但请记住,实际配置应根据具体需求进行微调。

11.使用 Spring Boot

如果我们使用 Spring Boot,我们可以使用更方便的方法来增加调度程序池的大小。
设置spring.task.scheduling.pool.size属性就足够了:
spring.task.scheduling.pool.size=5

posted @ 2024-03-07 21:39  dkpp  阅读(178)  评论(0编辑  收藏  举报