Daily timing task in system development are very common, for example, we might want to have a timer to query the status of a deal and summarize, we want to 4 o ‘clock in the morning or clear the related data of the database, or we want to zero at the end of a month time to open a transaction data statistics to make the report of monthly, quarterly, very much! I can’t finish!

A periodic task only tells the system to execute a task at a certain time. As for when the task will be completed, this is not the scope of the periodic task. A periodic task only needs to guarantee that a command to invoke a task will be issued at a certain time.

The leader in timed task implementation in Java is Quartz. The Quartz framework was widely used before Spring 3.0, and it still is, and it supports data persistence and cluster deployment. The spring version after Spring 3.0 implements a scheduled task framework of its own, which we can think of as an inferior version of Quartz. It does not provide clustering support, but other features are much more powerful.

In the spring tradition, spring-Task supports both XML and annotations to configure scheduled tasks. Of course, we will also explain how to customize scheduled tasks. In addition to spring-task, spring also provides quartz integration. For more information on how spring integrates Quartz, please read a short article about the Java timed task framework

I’ll explain how to use Spring-Task in terms of XML, annotations, and customization.

Implementing Spring scheduled tasks based on XML configuration

The scheduled task component is divided into three parts: scheduler, task, and execution point. We need to introduce three Spring packages: bean, Context, and core.

application.xml

<? xml version="1.0" encoding="UTF-8"? > <beans xmlns="http://www.springframework.org/schema/beans"    
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   
    xmlns:mvc="http://www.springframework.org/schema/mvc"
    xmlns:p="http://www.springframework.org/schema/p"  
    xmlns:aop="http://www.springframework.org/schema/aop"   
    xmlns:context="http://www.springframework.org/schema/context"  
    xmlns:jee="http://www.springframework.org/schema/jee"  
    xmlns:tx="http://www.springframework.org/schema/tx"  
    xmlns:task="http://www.springframework.org/schema/task"
    xsi:schemaLocation="Http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-4.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd"> <! Task: Scheduler id="schedualer"/ > <! Configure the task class bean --> <bean id="xmlTask" class="wokao666.club.spring_task.tasks.XmlTask"></bean>
	<task:scheduled-tasks scheduler="schedualer"> <! -- executed every 2 seconds --> < Task: Scheduled ref="xmlTask" method="say" cron="0/2 * * * *?"/>
	</task:scheduled-tasks>
</beans>
Copy the code

XmlTask.java

package wokao666.club.spring_task.tasks; import java.text.SimpleDateFormat; import wokao666.club.spring_task.util.DateFormatter; /** * Spring timed tasks based on XML */ public class XmlTask {public voidsay() {
        SimpleDateFormat format = DateFormatter.getDateFormatter();
        System.err.println(format.format(System.currentTimeMillis()) + " I am spring xml-based task!"); }}Copy the code

DateFormatter.java

package wokao666.club.spring_task.util; import java.text.SimpleDateFormat; @author HJW ** / Public class DateFormatter {private static Volatile SimpleDateFormat Formater = null; privateDateFormatter() {}
    public static SimpleDateFormat getDateFormatter() {
        if(null == formater) {
            synchronized (DateFormatter.class) {
                if(null == formater) {
                    formater = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss"); }}}returnformater; }}Copy the code

App.java

package wokao666.club.spring_task;

import org.springframework.context.support.ClassPathXmlApplicationContext;


/**
 * Hello world!
 *
 */
public class App 
{
    private static ClassPathXmlApplicationContext ctx;
    public static void main( String[] args ) {
            ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); }}Copy the code

Program output:

On June 10, 2018 afternoon 10:47:11 org. Springframework. Context. Support. AbstractApplicationContext prepareRefresh information: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@2328c243: startup date [Sun Jun 10 22:47:11 CST 2018]; Root of Context Hierarchy June 10, 2018 10:47:11 afternoon org. Springframework. Beans. Factory. XML. XmlBeanDefinitionReader loadBeanDefinitions information: Loading XML bean definitions from Class Path Resource [ApplicationContext.xml] 2018 10:47:12 afternoon org. Springframework. Scheduling. Concurrent. ExecutorConfigurationSupport initialize information: Initializing ExecutorService'schedualer'
2018-06-10 10:47:14 I am spring xml-based task!
2018-06-10 10:47:16 I am spring xml-based task!
2018-06-10 10:47:18 I am spring xml-based task!
Copy the code

Annotation-based Spring-Task

applicationContext.xml

<? xml version="1.0" encoding="UTF-8"? > <beans xmlns="http://www.springframework.org/schema/beans"    
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   
    xmlns:mvc="http://www.springframework.org/schema/mvc"
    xmlns:p="http://www.springframework.org/schema/p"  
    xmlns:aop="http://www.springframework.org/schema/aop"   
    xmlns:context="http://www.springframework.org/schema/context"  
    xmlns:jee="http://www.springframework.org/schema/jee"  
    xmlns:tx="http://www.springframework.org/schema/tx"  
    xmlns:task="http://www.springframework.org/schema/task"
    xsi:schemaLocation="Http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-4.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd"> <! Task :annotation-driven/> <task:annotation-driven/> <! -- bean scan --> <context:component-scan base-package="wokao666.club.spring_task.tasks"></context:component-scan>
</beans>
Copy the code

AnnotationTask.java

package wokao666.club.spring_task.tasks; import java.text.SimpleDateFormat; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Service; import wokao666.club.spring_task.util.DateFormatter; /** * Test annotation-based timed tasks * @author HJW */ @service public class AnnotationTask {@scheduled (cron=)"0/2 * * * *?")
    public void say() {
        SimpleDateFormat format = DateFormatter.getDateFormatter();
        System.err.println(format.format(System.currentTimeMillis()) + " good morning"); }}Copy the code

App. Java, dateFormatter. Java as above, the program output is as follows:

On June 10, 2018 afternoon 10:53:45 org. Springframework. Context. Support. AbstractApplicationContext prepareRefresh information: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@2328c243: startup date [Sun Jun 10 22:53:45 CST 2018]; Root of Context Hierarchy June 10, 2018 10:53:45 afternoon org. Springframework. Beans. Factory. XML. XmlBeanDefinitionReader loadBeanDefinitions information: Loading XML bean definitions from Class Path Resource [ApplicationContext.xml] 2018 10:53:46 afternoon org. Springframework. Beans. Factory. Support. DefaultListableBeanFactory registerBeanDefinition information: Overriding bean definitionfor bean 'org.springframework.context.annotation.internalScheduledAnnotationProcessor' with a different definition: replacing [Generic bean: class [org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null] with [Root bean: class [null]; scope=; abstract=false; lazyInit=false; autowireMode=3; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=org.springframework.scheduling.annotation.SchedulingConfiguration; factoryMethodName=scheduledAnnotationProcessor; initMethodName=null; destroyMethodName=(inferred); defined inThe class path resource [org/springframework/scheduling/annotation/SchedulingConfiguration. Class]] on June 10, 2018 10:53:46 afternoon org. Springframework. Scheduling. The annotation. ScheduledAnnotationBeanPostProcessor finishRegistration information: No TaskScheduler/ScheduledExecutorService bean foundfor scheduled processing
2018-06-10 10:53:48 good morning
2018-06-10 10:53:50 good morning
2018-06-10 10:53:52 good morning
2018-06-10 10:53:54 good morning
Copy the code

Note that the above output No TaskScheduler/ScheduledExecutorService bean found for scheduled processing this sentence, Because you can pass in a java.util.Executor thread pool object to implement calls to asynchronous tasks, if you specify the Executor property, Spring creates a SimpleAsyncTaskExecutor object to execute by default. The Executor property specifies the thread pool for scheduled task execution, emphasizing execution, and the Scheduler property specifies the thread pool for task scheduling, since Spring is single-threaded serial scheduling by default. To allow multi-threaded concurrent scheduling, the scheduler attribute needs to be configured as shown in the following example:

<! Task: Scheduler id="scheduler" pool-size="10"/ > <! -- Task execution pool --> <task:executor ID ="executor" pool-size="10" />
<task:annotation-driven executor="executor" scheduler="scheduler"/>
<context:component-scan base-package="wokao666.club.spring_task.tasks"></context:component-scan>
Copy the code

(It is necessary to further learn the relevant knowledge and principles of Spring asynchronous tasks, and use the @async annotation method for asynchronous execution.)

3. Customize scheduled tasks

Implementing custom Scheduled tasks, or the ability to change related CRon expressions, may seem like a crazy idea, but it is possible. Spring-@scheduled Scheduled tasks change cron parameters dynamically

CustomSchedual.java

package wokao666.club.spring_task.tasks; import org.springframework.scheduling.Trigger; import org.springframework.scheduling.annotation.EnableScheduling; import org.springframework.scheduling.annotation.SchedulingConfigurer; import org.springframework.scheduling.config.ScheduledTaskRegistrar; import org.springframework.scheduling.support.CronTrigger; import org.springframework.stereotype.Component; import wokao666.club.spring_task.util.DateFormatter; @enablesCheduling @Component Public Class CustomSchedual Implements SchedulingConfigurer{private static String cron ="0/2 * * * *?";
    
    private CustomSchedual() {
        new Thread(()->{
            try {
                Thread.sleep(16000);
            } catch (InterruptedException e) {}
            cron = "0/10 * * * *?";
            System.err.println("change")
        }).start();
    }
    public void configureTasks(ScheduledTaskRegistrar arg0) {
        arg0.addTriggerTask(()->{
            System.err.println(DateFormatter.getDateFormatter().format(System.currentTimeMillis()) + " good night!");
        },(triggerContext)->{
        CronTrigger trigger = new CronTrigger(cron);
        returntrigger.nextExecutionTime(triggerContext); }); }}Copy the code

The output of the program is as follows. As you can see, after 16 seconds of the program coming, the execution frequency changes from 2 seconds/time to 10 seconds/time:

On June 10, 2018 afternoon 11:49:13 org. Springframework. Context. Support. AbstractApplicationContext prepareRefresh information: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@2328c243: startup date [Sun Jun 10 23:49:13 CST 2018]; Root of Context Hierarchy June 10, 2018 11:49:13 afternoon org. Springframework. Beans. Factory. XML. XmlBeanDefinitionReader loadBeanDefinitions information: Loading XML bean definitions from Class Path Resource [ApplicationContext.xml] 2018 11:49:14 afternoon org. Springframework. Scheduling. The annotation. ScheduledAnnotationBeanPostProcessor finishRegistration information: No TaskScheduler/ScheduledExecutorService bean foundfor scheduled processing
2018-06-10 11:49:16 good night!
2018-06-10 11:49:18 good night!
2018-06-10 11:49:20 good night!
2018-06-10 11:49:22 good night!
2018-06-10 11:49:24 good night!
2018-06-10 11:49:26 good night!
2018-06-10 11:49:28 good night!
2018-06-10 11:49:30 good night!
change
2018-06-10 11:49:32 good night!
2018-06-10 11:49:40 good night!
2018-06-10 11:49:50 good night!
2018-06-10 11:50:00 good night!
2018-06-10 11:50:10 good night!
2018-06-10 11:50:20 good night!
Copy the code

Finally, simple use is not enough, we should not only know why, but also know why, later have time to learn the principle, in fact, the bottom is nothing more than the use of Java thread pool, come on, SAO years, good night!