数据库集群,怎么解决spring定时器的重复问题

数据库集群,怎么解决spring定时器的重复问题,第1张

有两种流行Spring定时器配置:Java的Timer类和OpenSymphony的Quartz。

1.Java Timer定时

首先继承java.util.TimerTask类实现run方法

import java.util.TimerTask

public class EmailReportTask extends TimerTask{

@Override

public void run() {

...

}

}

在Spring定义

...

配置Spring定时器

<bean id="scheduleReportTask" class="org.springframework.scheduling.timer.ScheduledTimerTask">

<property name="timerTask" ref="reportTimerTask" />

<property name="period">

<value>86400000value>

property>

bean>

timerTask属性告诉ScheduledTimerTask运行哪个。86400000代表24个小时

启动Spring定时器

Spring的TimerFactoryBean负责启动定时任务

<bean class="org.springframework.scheduling.timer.TimerFactoryBean">

<property name="scheduledTimerTasks">

<list><ref bean="scheduleReportTask"/>list>

property>

bean>

scheduledTimerTasks里显示一个需要启动的定时器任务的列表。

可以通过设置delay属性延迟启动

<bean id="scheduleReportTask" class="org.springframework.scheduling.timer.ScheduledTimerTask">

<property name="timerTask" ref="reportTimerTask" />

<property name="period">

<value>86400000value>

property>

<property name="delay">

<value>3600000value>

property>

bean>

这个任务我们只能规定每隔24小时运行一次,无法精确到某时启动

2.Quartz定时器

首先继承QuartzJobBean类实现executeInternal方法

import org.quartz.JobExecutionContext

import org.quartz.JobExecutionException

import org.springframework.scheduling.quartz.QuartzJobBean

public class EmailReportJob extends QuartzJobBean{

protected void executeInternal(JobExecutionContext arg0)

throws JobExecutionException {

...

}

}

在Spring中定义

<bean id="reportJob" class="org.springframework.scheduling.quartz.JobDetailBean">

<property name="jobClass">

<value>EmailReportJobvalue>

property>

<property name="jobDataAsMap">

<map>

<entry key="courseService">

<ref bean="courseService"/>

entry>

map>

property>

bean>

在这里我们并没有直接声明一个EmailReportJob Bean,而是声明了一个JobDetailBean。这个是Quartz的特点。JobDetailBean是Quartz的org.quartz.JobDetail的子类,它要求通过jobClass属性来设置一个Job对象。

使用Quartz的JobDetail中的另一个特别之处是EmailReportJob的courseService属性是间接设置的。JobDetail的jobDataAsMap属性接受一个Map,包括设置给jobClass的各种属性,当。JobDetailBean实例化时,它会将courseService Bean注入到EmailReportJob 的courseService 属性中。

启动定时器

Quartz的org.quartz.Trigger类描述了何时及以怎样的频度运行一个Quartz工作。Spring提供了两个触发器SimpleTriggerBean和CronTriggerBean。

SimpleTriggerBean与scheduledTimerTasks类似。指定工作的执行频度,模仿scheduledTimerTasks配置 .

<bean id="simpleReportTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean">

<property name="jobDetail" ref="reprotJob" />

<property name="startDelay">

<value>360000value>

property>

<property name="repeatInterval">

<value>86400000value>

property>

bean>

startDelay也是延迟1个小时启动

CronTriggerBean指定工作的准确运行时间

<bean id="cronReportTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">

<property name="jobDetail" ref="reprotJob" />

<property name="cronExpression">

<value>0 0 6 * * ?value>

property>

bean>

属性cronExpression告诉何时触发。最神秘就是cron表达式:

Linux系统的计划任务通常有cron来承担。一个cron表达式有至少6个(也可能7个)有空格分隔的时间元素。从左到右:

1.秒2.分3.小时4.月份中的日期(1-31)5.月份(1-12或JAN-DEC)6.星期中的日期(1-7或SUN-SAT)7.年份(1970-2099)

每个元素都显示的规定一个值(如6),一个区间(9-12),一个列表(9,11,13)或一个通配符(*)。因为4和6这两个元素是互斥的,因此应该通过设置一个问号(?)来表明不想设置的那个字段,“/”如果值组合就表示重复次数(10/6表示每10秒重复6次)。

启动定时器

<bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">

<property name="triggers">

<list><ref bean="cronReportTrigger"/>list>

property>

bean>

triggers属性接受一组触发器。

定时任务的实现方式有多种,例如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()

}

}

}


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/sjk/10064574.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-05-04
下一篇2023-05-04

发表评论

登录后才能评论

评论列表(0条)

    保存