shedlock源码解析
序
本文主要解析一下shedlock的实现。
LockProvider
shedlock-core-0.16.1-sources.jar!/net/javacrumbs/shedlock/core/LockProvider.java
public interface LockProvider {/*** @return If empty optional has been returned, lock could not be acquired. The lock* has to be released by the callee.*/Optional<SimpleLock> lock(LockConfiguration lockConfiguration);
}
LockProvider入参是lockConfiguration,返回SimpleLock。
StorageBasedLockProvider
shedlock-core-0.16.1-sources.jar!/net/javacrumbs/shedlock/support/StorageBasedLockProvider.java
public class StorageBasedLockProvider implements LockProvider {private final StorageAccessor storageAccessor;private final LockRecordRegistry lockRecordRegistry = new LockRecordRegistry();protected StorageBasedLockProvider(StorageAccessor storageAccessor) {this.storageAccessor = storageAccessor;}@Overridepublic Optional<SimpleLock> lock(LockConfiguration lockConfiguration) {boolean lockObtained = doLock(lockConfiguration);if (lockObtained) {return Optional.of(new StorageLock(lockConfiguration, storageAccessor));} else {return Optional.empty();}}/*** Sets lockUntil according to LockConfiguration if current lockUntil <= now*/protected boolean doLock(LockConfiguration lockConfiguration) {String name = lockConfiguration.getName();if (!lockRecordRegistry.lockRecordRecentlyCreated(name)) {// create document in case it does not exist yetif (storageAccessor.insertRecord(lockConfiguration)) {lockRecordRegistry.addLockRecord(name);return true;}lockRecordRegistry.addLockRecord(name);}return storageAccessor.updateRecord(lockConfiguration);}private static class StorageLock implements SimpleLock {private final LockConfiguration lockConfiguration;private final StorageAccessor storageAccessor;StorageLock(LockConfiguration lockConfiguration, StorageAccessor storageAccessor) {this.lockConfiguration = lockConfiguration;this.storageAccessor = storageAccessor;}@Overridepublic void unlock() {storageAccessor.unlock(lockConfiguration);}}}
使用StorageAccessor来实现加锁
LockManager
shedlock-core-0.16.1-sources.jar!/net/javacrumbs/shedlock/core/LockManager.java
/*** Executes task if not locked.*/
public interface LockManager {void executeWithLock(Runnable task);
}
默认实现
shedlock-core-0.16.1-sources.jar!/net/javacrumbs/shedlock/core/DefaultLockManager.java
public class DefaultLockManager implements LockManager {private static final Logger logger = LoggerFactory.getLogger(DefaultLockManager.class);private final LockingTaskExecutor lockingTaskExecutor;private final LockConfigurationExtractor lockConfigurationExtractor;public DefaultLockManager(LockProvider lockProvider, LockConfigurationExtractor lockConfigurationExtractor) {this(new DefaultLockingTaskExecutor(lockProvider), lockConfigurationExtractor);}public DefaultLockManager(LockingTaskExecutor lockingTaskExecutor, LockConfigurationExtractor lockConfigurationExtractor) {this.lockingTaskExecutor = requireNonNull(lockingTaskExecutor);this.lockConfigurationExtractor = requireNonNull(lockConfigurationExtractor);}@Overridepublic void executeWithLock(Runnable task) {Optional<LockConfiguration> lockConfigOptional = lockConfigurationExtractor.getLockConfiguration(task);if (!lockConfigOptional.isPresent()) {logger.debug("No lock configuration for {}. Executing without lock.", task);task.run();} else {lockingTaskExecutor.executeWithLock(task, lockConfigOptional.get());}}
}
委托给lockingTaskExecutor来加锁
shedlock-core-0.16.1-sources.jar!/net/javacrumbs/shedlock/core/DefaultLockingTaskExecutor.java
public class DefaultLockingTaskExecutor implements LockingTaskExecutor {private static final Logger logger = LoggerFactory.getLogger(DefaultLockingTaskExecutor.class);private final LockProvider lockProvider;public DefaultLockingTaskExecutor(LockProvider lockProvider) {this.lockProvider = requireNonNull(lockProvider);}@Overridepublic void executeWithLock(Runnable task, LockConfiguration lockConfig) {Optional<SimpleLock> lock = lockProvider.lock(lockConfig);if (lock.isPresent()) {try {logger.debug("Locked {}.", lockConfig.getName());task.run();} finally {lock.get().unlock();logger.debug("Unlocked {}.", lockConfig.getName());}} else {logger.debug("Not executing {}. It's locked.", lockConfig.getName());}}
}
这里跟lockProvider衔接上
SpringLockableTaskSchedulerFactoryBean(偷梁换柱
)
shedlock-spring-0.16.1-sources.jar!/net/javacrumbs/shedlock/spring/SpringLockableTaskSchedulerFactoryBean.java
@Overridepublic Class<?> getObjectType() {return LockableTaskScheduler.class;}@Overrideprotected LockableTaskScheduler createInstance() throws Exception {return new LockableTaskScheduler(taskScheduler,new DefaultLockManager(lockProvider, new SpringLockConfigurationExtractor(defaultLockAtMostFor, defaultLockAtLeastFor, embeddedValueResolver)));}
主要是LockableTaskScheduler的工厂方法
LockableTaskScheduler(task scheduler lock wrapper
)
shedlock-spring-0.16.1-sources.jar!/net/javacrumbs/shedlock/spring/LockableTaskScheduler.java
public class LockableTaskScheduler implements TaskScheduler, DisposableBean {private final TaskScheduler taskScheduler;private final LockManager lockManager;public LockableTaskScheduler(TaskScheduler taskScheduler, LockManager lockManager) {this.taskScheduler = requireNonNull(taskScheduler);this.lockManager = requireNonNull(lockManager);}@Overridepublic ScheduledFuture<?> schedule(Runnable task, Trigger trigger) {return taskScheduler.schedule(wrap(task), trigger);}@Overridepublic ScheduledFuture<?> schedule(Runnable task, Date startTime) {return taskScheduler.schedule(wrap(task), startTime);}@Overridepublic ScheduledFuture<?> scheduleAtFixedRate(Runnable task, Date startTime, long period) {return taskScheduler.scheduleAtFixedRate(wrap(task), startTime, period);}@Overridepublic ScheduledFuture<?> scheduleAtFixedRate(Runnable task, long period) {return taskScheduler.scheduleAtFixedRate(wrap(task), period);}@Overridepublic ScheduledFuture<?> scheduleWithFixedDelay(Runnable task, Date startTime, long delay) {return taskScheduler.scheduleWithFixedDelay(wrap(task), startTime, delay);}@Overridepublic ScheduledFuture<?> scheduleWithFixedDelay(Runnable task, long delay) {return taskScheduler.scheduleWithFixedDelay(wrap(task), delay);}private Runnable wrap(Runnable task) {return new LockableRunnable(task, lockManager);}@Overridepublic void destroy() throws Exception {if (taskScheduler instanceof DisposableBean) {((DisposableBean) taskScheduler).destroy();}}
}
对task scheduler包装了一层,织入了lock的逻辑
问题
上面将了半天,讲了lockProvider以及lockManager,还有LockableTaskScheduler是如何给task scheduler加上锁的,还有LockableTaskScheduler的工厂方法SpringLockableTaskSchedulerFactoryBean。那么问题来了,spring的schedule凭什么就使用你配置的LockableTaskScheduler呢?
@Beanpublic ScheduledLockConfiguration scheduledLockConfiguration(LockProvider lockProvider) {return ScheduledLockConfigurationBuilder.withLockProvider(lockProvider).withPoolSize(10).withDefaultLockAtMostFor(Duration.ofMinutes(10)).build();}
这种配置仅仅当spring工厂里头没有配置taskScheduler的时候,起作用。如果项目已经显示指定taskScheduler的时候,那么就不会使用LockableTaskScheduler。不过可以通过实现SchedulingConfigurer接口强制指定使用LockableTaskScheduler。
doc
使用shedlock将spring schedule上锁
shedlock源码解析相关推荐
- 谷歌BERT预训练源码解析(二):模型构建
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn.net/weixin_39470744/arti ...
- 谷歌BERT预训练源码解析(三):训练过程
目录 前言 源码解析 主函数 自定义模型 遮蔽词预测 下一句预测 规范化数据集 前言 本部分介绍BERT训练过程,BERT模型训练过程是在自己的TPU上进行的,这部分我没做过研究所以不做深入探讨.BE ...
- 谷歌BERT预训练源码解析(一):训练数据生成
目录 预训练源码结构简介 输入输出 源码解析 参数 主函数 创建训练实例 下一句预测&实例生成 随机遮蔽 输出 结果一览 预训练源码结构简介 关于BERT,简单来说,它是一个基于Transfo ...
- Gin源码解析和例子——中间件(middleware)
在<Gin源码解析和例子--路由>一文中,我们已经初识中间件.本文将继续探讨这个技术.(转载请指明出于breaksoftware的csdn博客) Gin的中间件,本质是一个匿名回调函数.这 ...
- Colly源码解析——结合例子分析底层实现
通过<Colly源码解析--框架>分析,我们可以知道Colly执行的主要流程.本文将结合http://go-colly.org上的例子分析一些高级设置的底层实现.(转载请指明出于break ...
- libev源码解析——定时器监视器和组织形式
我们先看下定时器监视器的数据结构.(转载请指明出于breaksoftware的csdn博客) /* invoked after a specific time, repeatable (based o ...
- libev源码解析——定时器原理
本文将回答<libev源码解析--I/O模型>中抛出的两个问题.(转载请指明出于breaksoftware的csdn博客) 对于问题1:为什么backend_poll函数需要指定超时?我们 ...
- libev源码解析——I/O模型
在<libev源码解析--总览>一文中,我们介绍过,libev是一个基于事件的循环库.本文将介绍其和事件及循环之间的关系.(转载请指明出于breaksoftware的csdn博客) 目前i ...
- libev源码解析——调度策略
在<libev源码解析--监视器(watcher)结构和组织形式>中介绍过,监视器分为[2,-2]区间5个等级的优先级.等级为2的监视器最高优,然后依次递减.不区分监视器类型和关联的文件描 ...
最新文章
- JSP实现大学生综合素质测评系统
- 苏州银行签约神策数据,致力打造科技引领的新时代普惠银行
- mysql查看执行计划_如何查看MySQL的执行计划
- 牛客题霸 [括号序列] C++题解/答案
- HDU 2056 Rectangles
- java class文件比较_java class文件查看工具
- 短信炸弹jmeter验证方法
- python 姓名转拼音
- 机器学习模型的超参数优化
- Windows创建快捷方式的几种方法你用过哪些?
- 实用小技巧 利用Python一秒将全部中文名转为拼音
- 社交巨人屏蔽Google搜索 Facebook发展强劲拟上市
- androidmmi可以卸载吗_删除Android自带的系统软件注意事项
- Modem analyze by T32 tool.
- 鼠标滚轮失灵上下乱跳的解决办法
- 无Internet访问权限-已解决
- 什么是Java SE、Java EE、Java ME?
- [MB855]变砖解决
- 新闻类客户端代码阅读笔记
- 校验码中码距与纠错能力的关系
热门文章
- 第 5 章 虚拟机栈
- 计算机维护宝典,超齐全的维修宝典之电脑维修实例大全
- moba寻路_MOBA代号105:道具收费 加入自动寻路等MMO元素
- beautifulsoup_如何使用 Python 和 BeautifulSoup 爬取网站
- matlab函数anova,MATLAB进行单因素方差分析-ANOVA
- leftjoin多表联合查询_结合mybatis-plus 实现无XML多表联合查询
- mysql innodb row_format_INNODB物理行结构(ROW_FORMAT=COMPACT) | 学步园
- python 进位_蓝桥杯-Python-高精度加法
- admin.php c install,laravel-admin后台的安装
- 腾讯云服务器CentOS 7安装MariaDB并用Navicat Premium连接