开发者

Spring ScheduledTask - start/stop support?

开发者 https://www.devze.com 2023-03-29 04:05 出处:网络
Is there a way to start or stop a task scheduled using Spring Scheduled Tasks initialized using context file or @Scheduled annotation?

Is there a way to start or stop a task scheduled using Spring Scheduled Tasks initialized using context file or @Scheduled annotation?

I would like to start the task when required and stop it when the task is no longer needed to be run.

If this is not possible, any alternative to injecting spring variables开发者_StackOverflow中文版 to a thread?


Here is an example of starting/stopping a scheduled method in Spring Boot. You can use such APIs:
http:localhost:8080/start - for starting scheduled method with fixed rate 5000 ms
http:localhost:8080/stop - for stopping scheduled method

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import java.time.Instant;
import java.util.concurrent.ScheduledFuture;

@Configuration
@ComponentScan
@EnableAutoConfiguration    
public class TaskSchedulingApplication {

    public static void main(String[] args) {
        SpringApplication.run(TaskSchedulingApplication.class, args);
    }

    @Bean
    TaskScheduler threadPoolTaskScheduler() {
        return new ThreadPoolTaskScheduler();
    }
}

@Controller
class ScheduleController {

    public static final long FIXED_RATE = 5000;

    @Autowired
    TaskScheduler taskScheduler;

    ScheduledFuture<?> scheduledFuture;

    @RequestMapping("start")
    ResponseEntity<Void> start() {
        scheduledFuture = taskScheduler.scheduleAtFixedRate(printHour(), FIXED_RATE);

        return new ResponseEntity<Void>(HttpStatus.OK);
    }

    @RequestMapping("stop")
    ResponseEntity<Void> stop() {
        scheduledFuture.cancel(false);
        return new ResponseEntity<Void>(HttpStatus.OK);
    }

    private Runnable printHour() {
        return () -> System.out.println("Hello " + Instant.now().toEpochMilli());
    }

}


Stopping registered @Scheduled beans is not standard feature since the access to them is private in the org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor.

If you need to manage the time they run you need to register them programmatically (TriggerTask): see the documentation to org.springframework.scheduling.annotation.SchedulingConfigurer. In the type org.springframework.scheduling.config.TriggerTask there is the method which returns type of org.springframework.scheduling.Trigger. There you can manage next execution time.

TriggerTasks could be beans in the case of programmatical registration.


Have the @Scheduled method look for a variable held in Application state or ServletContext, or from a value stored in the DB. If the value is TRUE, proceed with the task; if FALSE, don't start. This setup will control the scheduled run.

If you want to also be able to fire the task at will, reference the task's method from a Controller; that way you can fire it at will. Additionally, if its a longer running task, create a second method annotated @Async and call that method from your Controller so that it runs in its own thread.


There is a pretty simple way of doing it (Spring Boot 2.3.12.RELEASE). I have a rule that all my job classes must implement Runnable and run method has @Scheduled annotation. Method proceed returns a new ScheduledTask object that can be suspended. Please note that postProcessBeforeDestruction removes "suspended" scheduled tasks from internal map hence we need to create a new one by calling postProcessAfterInitialization

    @RequiredArgsConstructor
    public class CustomScheduledTaskRegistrar extends ScheduledTaskRegistrar {

        private final JobTrackingManager jobTrackingManager; // Custom class to keep records on ScheduledTask objects

        @Override
        public void afterPropertiesSet() {
            super.afterPropertiesSet();
            jobTrackingManager.registerScheduledTasks();
        }

    } 


    public class CustomScheduledAnnotationBeanPostProcessor extends ScheduledAnnotationBeanPostProcessor {

        public CustomScheduledAnnotationBeanPostProcessor(CustomScheduledTaskRegistrar customScheduledTaskRegistrar) {
            super(customScheduledTaskRegistrar);
        }

        public void suspend(ScheduledTask scheduledTask) {
            Object bean = ((ScheduledMethodRunnable) scheduledTask.getTask().getRunnable()).getTarget();
            postProcessBeforeDestruction(bean, "Ignored");
        }

        @SneakyThrows
        public ScheduledTask proceed(ScheduledTask scheduledTask) {
            Object bean = ((ScheduledMethodRunnable) scheduledTask.getTask().getRunnable()).getTarget();
            if (!requiresDestruction(bean)) {
                String beanName = AopProxyUtils.ultimateTargetClass(bean).getSimpleName();
                postProcessAfterInitialization(bean, beanName);
            }
            return getScheduledTasks().stream()
            .filter(st -> ((ScheduledMethodRunnable) st.getTask().getRunnable()).getTarget().equals(bean))
            .findFirst()
            .orElseThrow(() -> new IllegalStateException("Unable to find the scheduled task"));
        }

    }
0

精彩评论

暂无评论...
验证码 换一张
取 消