scheduledtask(Scheduled 定时任务)
一、定时任务 | @scheduled
SpringBoot为我们内置了@Scheduled定时任务,下面我们就来配置下这个注解,找到入口程序添加注解@EnableScheduling,添加注解后SpringBoot就已经认定了我们要使用定时任务来完成一些业务逻辑了,内部会对应原始配置定时任务添加对应的配置文件
@scheduled注解用来配置到方法上来完成对应的定时任务的配置,如执行时间,间隔时间,延迟时间等等,下面我们就来详细的看下对应的属性配置
下面我们来配置@Scheduled,来完成每小时输出内容
Seconds:可出现",-*/"四个字符,有效范围为0-59的整数
Minutes:可出现",-*/"四个字符,有效范围为0-59的整数
Hours:可出现",-*/"四个字符,有效范围为0-23的整数
DayofMonth:可出现",-*/? L W C"八个字符,有效范围为0-31的整数
Month:可出现",-*/"四个字符,有效范围为1-12的整数或JAN-DEc
DayofWeek:可出现",-*/? L C#"四个字符,有效范围为1-7的整数或SUN-SAT两个范围。1表示星期天,2表示星期一,依次类推
Year:可出现",-*/"四个字符,有效范围为1970-2099年
下面简单举几个例子:
"0 0 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:55分结束每5分钟一次触发
"0 0/5 14,18**?"每天的下午2点至2:55和6点至6点55分两个时间段内每5分钟一次触发
"0 0-5 14**?"每天14:00至14:05每分钟一次触发
"0 10,44 14? 3 WED"三月的每周三的14:10和14:44触发
"0 15 10?* MON-FRI"每个周一、周二、周三、周四、周五的10:15触发
该属性的含义是指在项目启动后在定义的时间后开启定时任务,配合fixedRate()和fixedDelay()使用
该属性的含义是调用固定周期(以毫秒为单位)执行方法:就是上一次开始执行时间点之后延迟执行
该属性的含义是上次调用结束和下一次调用结束之间的固定周期(以毫秒为单位)执行方法:就是上一次执行完毕时间点之后延迟执行。
1、cron、fixedDelay、fixedRate三者之间不能共存!!!
2、fixedDelay、fixedRate不能单独使用!!!
会抛出一个错误:
二、Spring使用@Scheduled进行定时任务,定的时间可否变
定时任务的实现方式有多种,例如JDK自带的Timer+TimerTask方式,Spring 3.0以后的调度任务(Scheduled Task),Quartz等。
Timer+TimerTask是最基本的解决方案,但是比较远古了,这里不再讨论。Spring自带的Scheduled
Task是一个轻量级的定时任务调度器,支持固定时间(支持cron表达式)和固定时间间隔调度任务,支持线程池管理。以上两种方式有一个共同的缺点,那就是应用服务器集群下会出现任务多次被调度执行的情况,因为集群的节点之间是不会共享任务信息的,每个节点上的任务都会按时执行。Quartz是一个功能完善的任务调度框架,特别牛叉的是它支持集群环境下的任务调度,当然代价也很大,需要将任务调度状态序列化到数据库。Quartz框架需要10多张表协同,配置繁多,令人望而却步...
经过折中考虑,还是选择了Spring的Scheduled Task来实现定时任务。如下:
1. Spring配置文件application-context.xml中添加task命名空间和描述。
[html] view plain copy
<beans xmlns=""
xmlns:task=""
xsi:schemaLocation="
/spring-beans.xsd
/spring-task.xsd">
2.添加调度器和线程池声明。
[html] view plain copy
<task:executor id="taskExecutor" pool-size="10"/>
<task:annotation-driven executor="taskExecutor"/>
3.实现调度方法。基本结构如下:
[html] view plain copy
package com.netease.yx.service;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;
@Service
public class ScheduledService{
@Scheduled(cron="0 0 5***")
public void build(){
System.out.println("Scheduled Task");
}
}
@Scheduled注解支持秒级的cron表达式,上述声明表示每天5点执行build任务。
前文已经提过,这种方式在单台应用服务器上运行没有问题,但是在集群环境下,会造成build任务在5点的时候运行多次,遗憾的是,Scheduled Task在框架层面没有相应的解决方案,只能靠程序员在应用级别进行控制。
如何控制看
1.无非是一个任务互斥访问的问题,声明一把全局的逗锁地作为互斥量,哪个应用服务器拿到这把逗锁地,就有执行任务的权利,未拿到逗锁地的应用服务器不进行任何任务相关的操作。
2.这把逗锁地最好还能在下次任务执行时间点前失效。
在项目中我将这个互斥量放在了redis缓存里,1小时过期,这个过期时间是由任务调度的间隔时间决定的,只要小于两次任务执行时间差,大于集群间应用服务器的时间差即可。
完整定时任务类如下:
[html] view plain copy
package com.netease.yx.service;
import javax.annotation.Resource;
import org.apache.commons.lang3.time.DateUtils;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;
import com.netease.yx.service.ICacheService;
@Service
public class ScheduledService{
@Resource
private ICacheService cache= null;
private static String CACHE_LOCK="cache_lock";
private static int EXPIRE_PERIOD=(int)DateUtils.MILLIS_PER_HOUR/ 1000;
@Scheduled(cron="0 0 5***")
public void build(){
if(cache.get(CACHE_LOCK)== null){
cache.set(CACHE_LOCK, true, EXPIRE_PERIOD);
doJob();
}
}
}
三、Spring 定时任务 @Scheduled cron表达式
我们在开发时经常会遇到一些需要定时执行的小任务,使用了 springboot的定时任务后变得更加简单快捷,下面举个例子:
Java配置中开户对Scheduled的支持:
Schedule定时器cron表达式:
Cron表达式是一个字符串,字符串以5或6个空格隔开,分为6或7个域,每一个域代表一个含义,Cron有如下两种语法格式:
一个cron表达式由空格分隔6个或者7个占位符组成,每个占位符代表不同意义,分别为:秒、分钟、小时、日、月、周、年
*:代表整个时间段。假如在Minutes域使用*,即表示每分钟都会触发事件。
?:表示每月的某一天,或第周的某一天;只能用在DayofMonth和DayofWeek两个域,表示不指定值。当2个子表达式其中之一被指定了值以后,为了避免冲突,需要将另一个子表达式的值设为“?”。例如想在每月的20日触发调度,不管20日到底是星期几,则只能使用如下写法: 13 13 15 20*?,其中最后一位只能用?,而不能使用,如果使用表示不管星期几都会触发,实际上并不是这样。
-:表示范围,例如在Minutes域使用 5-20,表示从5分到20分钟每分钟触发一次。
/:表示起始时间开始触发,然后每隔固定时间触发一次。例如在Minutes域使用“0/10”,表示每隔10分钟执行一次,“0”表示为从“0”分开始;如果是“5/20”,则意味着从5分开始,每20分钟触发一次,而25,45等分别触发一次,“5”表示从第5分钟开始执行.。
,:表示列出枚举值值。例如:在Minutes域使用5,20,则意味着在第5分和第20分各触发一次。
L:表示最后,只能出现在DayofWeek和DayofMonth域,用于每月,或每周,表示为每月的最后一天,或每个月的最后星期几。如果在DayofWeek域使用“5L”,意味着在最后的一个星期四触发;如“6L”表示“每月的最后一个星期五”。
W:表示有效工作日(周一到周五),只能出现在DayofMonth域,系统将在离指定日期的最近的有效工作日触发事件。例如:在 DayofMonth使用“5W”表示为“到本月5日最近的工作日”,如果5日是星期六,则将在最近的工作日:星期五,即4日触发。如果5日是星期天,则在6日(周一)触发;如果5日在星期一到星期五中的一天,则就在5日触发。另外一点,W的最近寻找不会跨过月份。
LW:这两个字符可以连用,表示在某个月最后一个工作日,即最后一个星期五。
#:用于确定每个月第几个星期几,只能出现在DayofMonth域。例如在4#2,表示每月的第二个星期三;如果是”6#3”或“FRI#3”,则表示“每月第三个星期五”。
最后推荐一个在线cron表达式生成器: