1. 程式人生 > >Spring的@Scheduled任務排程

Spring的@Scheduled任務排程

一. 定時任務實現方式

定時任務實現方式:

  • Java自帶的java.util.Timer類,這個類允許你排程一個java.util.TimerTask任務。使用這種方式可以讓你的程式按照某一個頻度執行,但不能在指定時間執行。一般用的較少,這篇文章將不做詳細介紹。
  • 使用Quartz,這是一個功能比較強大的的排程器,可以讓你的程式在指定時間執行,也可以按照某一個頻度執行,配置起來稍顯複雜,有空介紹。
  • 使用Spring的@Scheduled註解配合@EnableScheduling一起使用。
  • SpringBoot自帶的Scheduled,可以將它看成一個輕量級的Quartz,而且使用起來比Quartz簡單許多,本文主要介紹。

說明:@Scheduled 註解用於標註這個方法是一個定時任務的方法,有多種配置可選。cron支援cron表示式,指定任務在特定時間執行;fixedRate以特定頻率執行任務;fixedRateString以string的形式配置執行頻率。

定時任務執行方式:

  • 單執行緒(序列)
  • 多執行緒(並行)

二. 建立定時任務

2.1、Spring boot中序列排程

在Spring boot中自帶了Seheduled,實現起來很方便,只需要在需要排程的方法上增加註解即可。如下:

package com.dxz.demo.schedule;
import java.time.LocalDateTime;

import org.springframework.scheduling.annotation.EnableScheduling; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Service; @Service //@EnableScheduling 可以省略 public class PrintJob { @Scheduled(initialDelay=3000, fixedDelay = 10000) public void
print() { Thread current = Thread.currentThread(); System.out.println(LocalDateTime.now() + " thread-name:" + current.getName() + ": 60 print"); } }

結果:

2.2、Spring boot中並行排程

繼承SchedulingConfigurer類並重寫其方法即可,如下:

package com.dxz.demo.schedule;

import java.util.concurrent.Executor;
import java.util.concurrent.Executors;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;

@Configuration
@EnableScheduling
public class ScheduleConfig implements SchedulingConfigurer {

    @Override
    public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
        taskRegistrar.setScheduler(taskExecutor());
    }

    @Bean(destroyMethod = "shutdown")
    public Executor taskExecutor() {
        return Executors.newScheduledThreadPool(10);
    }
}

結果:

多個執行緒依次按照固定頻率來執行排程的。

2.3、Spring boot中非同步並行排程

2.4、Spring並行排程

1.新建一個web工程,引入spring相關包,pom檔案如下

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>com.dxz.demo</groupId>
    <artifactId>SpringDemo</artifactId>
    <packaging>war</packaging>
    <version>0.0.1-SNAPSHOT</version>
    <name>SpringDemo Maven Webapp</name>
    <url>http://maven.apache.org</url>
    <dependencies>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>4.1.7.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-web</artifactId>
            <version>4.1.6.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-webmvc</artifactId>
            <version>4.1.6.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>3.8.1</version>
            <scope>test</scope>
        </dependency>
    </dependencies>
    <build>
        <finalName>SpringDemo</finalName>
    </build>
</project>

spring配置檔案

<?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:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
http://www.springframework.org/schema/beans/spring-beans.xsd 
http://www.springframework.org/schema/context 
http://www.springframework.org/schema/context/spring-context.xsd">

    <import resource="applicationContext-task.xml" />
    <context:component-scan base-package="com.dxz.demo" />
</beans>

<?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:context="http://www.springframework.org/schema/context"
    xmlns:task="http://www.springframework.org/schema/task"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd 
    http://www.springframework.org/schema/context 
    http://www.springframework.org/schema/context/spring-context.xsd 
    http://www.springframework.org/schema/task 
    http://www.springframework.org/schema/task/spring-task.xsd">

    <task:annotation-driven executor="taskExecutor"
        scheduler="taskScheduler" />
    <task:executor id="taskExecutor" pool-size="5" />
    <task:scheduler id="taskScheduler" pool-size="10" />
</beans>

web.xml配置

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns="http://java.sun.com/xml/ns/javaee"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
    id="WebApp_ID" version="3.0">
    <display-name>appversion</display-name>
    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:applicationContext.xml</param-value>
    </context-param>
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

</web-app>

排程任務類

package com.dxz.demo;

import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

@Component
public class HelloWorldTask {
    @Scheduled(fixedRate = 5000)
    public void doSomething() throws Exception {
        System.out.println(Thread.currentThread().getName() + " doSomething is invoked!");
    }

    @Scheduled(fixedDelay = 5000)
    public void doSomething2() throws Exception {
        System.out.println(Thread.currentThread().getName() + " doSomething2 is invoked!");
    }
}

工程結構如下

結果:

從上圖中可以看到時多個執行緒在執行排程任務的。

三. @Scheduled引數說明:

initial-delay : 表示第一次執行前需要延遲的時間,單位是毫秒
fixed-delay : 表示從上一個任務完成到下一個任務開始的間隔, 單位是毫秒。
fixed-rate : 表示從上一個任務開始到下一個任務開始的間隔, 單位是毫秒。(如果上一個任務執行超時,則可能是上一個任務執行完成後立即啟動下一個任務)
cron : cron 表示式。(定時執行,如果上一次任務執行超時而導致某個定時間隔不能執行,則會順延下一個定時間隔時間。下一個任務和上一個任務的間隔時間不固定)
區別見圖:

4、侷限性——@Scheduled的cron無法指定執行的年份

  即我們假如使用下面的定時任務

    @Scheduled(cron = "0 18 10 * * ? 2016-2016")
    public void testTaskWithDate() {
        logger.info("測試2016.定時任務");
    }

   將會報下面的錯誤

Cron expression must consist of 6 fields (found 7 in "0 18 10 * * ? 2016-2016")
Caused by: java.lang.IllegalStateException: Encountered invalid @Scheduled method 'testTaskWithDate': Cron expression must consist of 6 fields (found 7 in "0 18 10 * * ? 2016-2016")
    at org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor.processScheduled(ScheduledAnnotationBeanPostProcessor.java:405)
    at org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor.postProcessAfterInitialization(ScheduledAnnotationBeanPostProcessor.java:258)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.applyBeanPostProcessorsAfterInitialization(AbstractAutowireCapableBeanFactory.java:422)

錯誤原因:

複製程式碼
/**
 * Parse the given pattern expression.
 */
private void parse(String expression) throws IllegalArgumentException {
    String[] fields = StringUtils.tokenizeToStringArray(expression, " ");
    if (fields.length != 6) {
        throw new IllegalArgumentException(String.format(""
                + "cron expression must consist of 6 fields (found %d in %s)", fields.length, expression));
    }
複製程式碼

  spring taks 不支援年位定時,它畢竟不是quartz,只是簡單的定時框架,比起jdk Timer就加入了執行緒池而以.

  但是制定到年份,會存在一個問題,就是在你在過了這個時間後再啟動專案的時候,他會一直報一個memory leak的錯誤,大概的意思是你的定時任務將永遠不會被執行,導致專案一直啟動不了。

原始碼裡的註釋:

*The pattern is a list of six single space-separated fields: representing * second, minute, hour, day, month, weekday. Month and weekday names can be * given as the first three letters of the English names.

四、Cron表示式的詳細用法

欄位 允許值 允許的特殊字元
秒 0-59 , - * /
分 0-59 , - * /
小時 0-23 , - * /
日期 1-31 , - * ? / L W C
月份 1-12 或者 JAN-DEC , - * /
星期 1-7 或者 SUN-SAT , - * ? / L C #
年(可選) 留空, 1970-2099 , - * /
例子:
0/5 * * * * ? : 每5秒執行一次
“*”字元被用來指定所有的值。如:"*"在分鐘的欄位域裡表示“每分鐘”。
“?”字元只在日期域和星期域中使用。它被用來指定“非明確的值”。當你需要通過在這兩個域中的一個來指定一些東西的時候,它是有用的。看下面的例子你就會明白。
月份中的日期和星期中的日期這兩個元素時互斥的一起應該通過設定一個問號來表明不想設定那個欄位。
“-”字元被用來指定一個範圍。如:“10-12”在小時域意味著“10點、11點、12點”。
“,”字元被用來指定另外的值。如:“MON,WED,FRI”在星期域裡表示”星期一、星期三、星期五”。
“/”字元用於指定增量。如:“0/15”在秒域意思是每分鐘的0,15,30和45秒。“5/15”在分鐘域表示每小時的5,20,35和50。 符號“*”在“/”前面(如:*/10)等價於0在“/”前面(如:0/10)。記住一條本質:表示式的每個數值域都是一個有最大值和最小值的集合,如: 秒域和分鐘域的集合是0-59,日期域是1-31,月份域是1-12。字元“/”可以幫助你在每個字元域中取相應的數值。如:“7/6”在月份域的時候只 有當7月的時候才會觸發,並不是表示每個6月。
L是‘last’的省略寫法可以表示day-of-month和day-of-week域,但在兩個欄位中的意思不同,例如day-of- month域中表示一個月的最後一天。如果在day-of-week域表示‘7’或者‘SAT’,如果在day-of-week域中前面加上數字,它表示 一個月的最後幾天,例如‘6L’就表示一個月的最後一個星期五。
字元“W”只允許日期域出現。這個字元用於指定日期的最近工作日。例如:如果你在日期域中寫 “15W”,表示:這個月15號最近的工作日。所以,如果15號是週六,則任務會在14號觸發。如果15好是週日,則任務會在週一也就是16號觸發。如果 是在日期域填寫“1W”即使1號是週六,那麼任務也只會在下週一,也就是3號觸發,“W”字元指定的最近工作日是不能夠跨月份的。字元“W”只能配合一個 單獨的數值使用,不能夠是一個數欄位,如:1-15W是錯誤的。
“L”和“W”可以在日期域中聯合使用,LW表示這個月最後一週的工作日。
字元“#”只允許在星期域中出現。這個字元用於指定本月的某某天。例如:“6#3”表示本月第三週的星期五(6表示星期五,3表示第三週)。“2#1”表示本月第一週的星期一。“4#5”表示第五週的星期三。
字元“C”允許在日期域和星期域出現。這個字元依靠一個指定的“日曆”。也就是說這個表示式的值依賴於相關的“日曆”的計算結果,如果沒有“日曆” 關聯,則等價於所有包含的“日曆”。如:日期域是“5C”表示關聯“日曆”中第一天,或者這個月開始的第一天的後5天。星期域是“1C”表示關聯“日曆” 中第一天,或者星期的第一天的後1天,也就是週日的後一天(週一)。

五、表示式舉例

"0 0 12 * * ?" 每天中午12點觸發
"0 15 10 ? * *" 每天上午10:15觸發
"0 15 10 * * ?" 每天上午10:15觸發
"0 15 10 * * ? *" 每天上午10:15觸發
"0 15 10 * * ? 2005" 2005年的每天上午10:15觸發
"0 * 14 * * ?" 在每天下午2點到下午2:59期間的每1分鐘觸發
"0 0/5 14 * * ?" 在每天下午2點到下午2:55期間的每5分鐘觸發
"0 0/5 14,18 * * ?" 在每天下午2點到2:55期間和下午6點到6:55期間的每5分鐘觸發
"0 0-5 14 * * ?" 在每天下午2點到下午2:05期間的每1分鐘觸發
"0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44觸發
"0 15 10 ? * MON-FRI" 週一至週五的上午10:15觸發
"0 15 10 15 * ?" 每月15日上午10:15觸發
"0 15 10 L * ?" 每月最後一日的上午10:15觸發
"0 15 10 ? * 6L" 每月的最後一個星期五上午10:15觸發
"0 15 10 ? * 6L 2002-2005" 2002年至2005年的每月的最後一個星期五上午10:15觸發
"0 15 10 ? * 6#3" 每月的第三個星期五上午10:15觸發

        
常用示例:

0 0 12 * * ? 每天12點觸發
0 15 10 ? * * 每天10點15分觸發
0 15 10 * * ? 每天10點15分觸發
0 15 10 * * ? * 每天10點15分觸發
0 15 10 * * ? 2005 2005年每天10點15分觸發
0 * 14 * * ? 每天下午的 2點到2點59分每分觸發
0 0/5 14 * * ? 每天下午的 2點到2點59分(整點開始,每隔5分觸發)
0 0/5 14,18 * * ? 每天下午的 2點到2點59分、18點到18點59分(整點開始,每隔5分觸發)
0 0-5 14 * * ? 每天下午的 2點到2點05分每分觸發
0 10,44 14 ? 3 WED 3月分每週三下午的 2點10分和2點44分觸發
0 15 10 ? * MON-FRI 從週一到週五每天上午的10點15分觸發
0 15 10 15 * ? 每月15號上午10點15分觸發
0 15 10 L * ? 每月最後一天的10點15分觸發
0 15 10 ? * 6L 每月最後一週的星期五的10點15分觸發
0 15 10 ? * 6L 2002-2005 從2002年到2005年每月最後一週的星期五的10點15分觸發
0 15 10 ? * 6#3 每月的第三週的星期五開始觸發
0 0 12 1/5 * ? 每月的第一個中午開始每隔5天觸發一次
0 11 11 11 11 ? 每年的11月11號 11點11分觸發(光棍節)

參考:http://wujq4java.iteye.com/blog/2067214

參考:http://blog.csdn.net/applebomb/article/details/52400154

相關推薦

Spring的@Scheduled任務排程

一. 定時任務實現方式 定時任務實現方式: Java自帶的java.util.Timer類,這個類允許你排程一個java.util.TimerTask任務。使用這種方式可以讓你的程式按照某一個頻度執行,但不能在指定時間執行。一般用的較少,這篇文章將不做詳細介紹。 使用Quartz,這是一個功能比較強

Spring定時任務排程scheduled-tasks>【含cron引數詳解】

spring內部有一個task是Spring自帶的一個設定時間自動任務排程task使用的時候很方便,但是他能做的東西不如quartz那麼的多!可以使用註解和配置兩種方式,配置的方式如下引入Spring放在appcation.xml開頭<beans xmlns="http

spring定時任務排程quartz

1.maven依賴 <!-- 定時任務 --> <dependency> <groupId>quartz</groupId> <artifactId>quartz</artifactId> <v

Spring任務排程之task:scheduler與task:executor配置的詳解

其實就是Spring定時器中配置檔案中一些配置資訊,由於筆者自己是頭一次使用,有些配置詳細不太明白,隨即研究了一番,於是想記錄一下,有需要的小夥伴可以參考,也方便日後自己查閱。 首先,建立一個僅僅包含定時器配置的Spring配置檔案:spring-timer.xml。以下均為配置資訊:1、在配置檔案

spring boot 任務排程

1.依賴檔案 檔名:pom.xml 檔案內容: <?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0"      

spring task 任務排程 (定時任務

0 0 10,14,16 * * ? 每天上午10點,下午2點,4點 0 0/30 9-17 * * ?   朝九晚五工作時間內每半小時 0 0 12 ? * WED 表示每個星期三中午12點  "0 0 12 * * ?" 每天中午12點觸發  "0 15 10 ? * *" 每天上午10:15觸發  "0

spring自動任務排程

Spring內部有一個task是Spring自帶的一個設定時間自動任務排程 task使用的時候很方便,但是他能做的東西不如quartz那麼的多! 可以使用註解和配置兩種方式,配置的方式如下 引入Spring放在appcation.xml開頭 <bean

轉:Spring任務排程cronExpression配置說明

cronExpression配置說明 欄位 允許值 允許的特殊字元 秒 0-59 , - * / 分 0-59 , - * / 小時 0-23 , - * / 日期 1-31 , - * ? / L W C 月份

spring定時任務排程

本文將告訴你如何使用spring的任務排程。主要使用@Scheduled註解 需要會使用maven 第一步 pom.xml配置 <?xml version="1.0" encoding="UTF-8"?> <proje

Quartz Job & Spring 動態任務排程

Quartz Job & Spring 在實際專案應用中經常會用到定時任務,可通過Quartz框架輕鬆完成。在Web專案中,如果用Spring框架管理Quartz,在Web容器啟動或關閉時自動啟動、關閉Quartz中的任務,非常方便。 傳統的Meth

Spring之——兩種任務排程Scheduled和Async

轉載請註明出處:http://blog.csdn.net/l1028386804/article/details/72494169 1、Spring排程的兩種方式 Spring提供了兩種後臺任務的方法,分別是:     排

Spring 定時任務之 @Scheduled cron表達式

按順序 cron sun ron cell last div text table 一個cron表達式有至少6個(也可能7個)有空格分隔的時間元素。 按順序依次為 秒(0~59) 分鐘(0~59) 小時(0~23) 天(月)(0~31,但是你需要考慮你月的天數) 月(0~

Spring 定時任務Scheduled 開發詳細圖文

定時執行 本地 setting 工具類 location fir clean scan crontab Spring 定時任務Scheduled 開發 文章目錄 一、前言 1.1 定時任務 1.2 開發環境 1.3 技術實現 二、創建包含WEB.

Spring整合quartz定時任務排程的cronExpression配置說明(轉載)

Spring整合quartz定時任務排程 "* * * * * *" 欄位   允許值   允許的特殊字元 秒   

spring定時任務@Scheduled,非同步操作@Async

需求時定時更新專案裡面某一個裝置的狀態。 1.定時任務:spring定時任務@Scheduled(cron = "50 * * * * ? ") 2.更新狀態採用非同步更新,java預設是同步的,非同步採用spring的@Async("async_update_gbStatus")

Spring任務排程實戰之Quartz Simple Trigger

分享一下我老師大神的人工智慧教程!零基礎,通俗易懂!http://blog.csdn.net/jiangjunshow 也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!        

Spring的定時任務任務排程)兩種

Spring內部有一個task是Spring自帶的一個設定時間自動任務排程,提供了兩種方式進行配置,一種是註解的方式,而另外一種就是XML配置方式了。註解方式比較簡潔,XML配置方式相對而言有些繁瑣,但是應用場景的不同,兩者又各有優點,所以具體使用還是根據需求來劃分。因為任務排程這樣的需求,

Spring的業務層和Web層-----------任務排程

任務排程   quartz框架   quartz框架實現了Spring的任務排程,使用者可以隨意的定義觸發器排程時間表,並將觸發器和任務進行對映。quartz通過排程器、觸發器和任務實現任務排程。   Job:主要用來設計任務實現的邏輯,並且只有一個方法execute。   JobDetail:主要用

一張圖讓你秒懂Spring @Scheduled定時任務的fixedRate,fixedDelay,cron執行差異

https://blog.csdn.net/applebomb/article/details/52400154   看字面意思容易理解,但是任務執行長度超過週期會怎樣呢? 不多說,直接上圖: 測試程式碼: import java.text.DateFormat; imp

Spring定時任務的@Scheduled cron表示式

目錄 一、文章前言     在@Scheduled的cron表示式中通常有6/7個屬性,今天上網檢視模式的時候發現很多的參考文章都有錯誤,最常見的就是天數的部分為0-31,月份等,文章前後對應不起來,在使用上造成了一定的困擾,所以特意寫了一份參考,改正了其