在项目中使用@Scheduled注解实现定时调度十分方便,但是存在一个情况:当项目多节点部署时,调度会在每个节点上都执行。这就是个大问题了。网上解决方案也比较多,我用过两种方案:第一种是:指定一台主机,让调度只在一台主机上执行。第二种是:使用redis的分布式锁实现。
第一种,失去了分布式部署的宗旨。下面我简单的介绍下第二种
1、创建Redis工具类中,增加setNX方法,即"SET if Not Exists",成功返回true,失败返回flase:
public Boolean setNX(String key, String value,long timeout, TimeUnit unit) {
Boolean isExit = this.redisTemplate.getConnectionFactory().getConnection().setNX(key.getBytes(), value.getBytes());
//如果设置成功,要设置其过期时间
if (isExit) {
redisTemplate.expire(key, timeout, unit);
}
return isExit;
}
2、在业务代码中引用方法:
public void testtest() throws InterruptedException {
try{
if (!redisUtil.setNX("quartzFlag","1",30,TimeUnit.SECONDS)){//设置超时时间
LOGGER.info("任务已执行");
return;
}
业务代码....
...........
} catch (Exception e) {
e.printStackTrace();
} finally {
//任务完成后,释放redis锁
RedisUtil.del("activityScheduledFlag");
}
为防止程序意外终止,导致死锁,文章中用到了,任务完成释放锁和设置过期时间双重保障。
这是在使用的时候总结的。如有错误,请批评指正。
一切的真理都是实践中产生的。交给时间去验证吧