本文主要解析一下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 &lt;= 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源码解析相关推荐

  1. 谷歌BERT预训练源码解析(二):模型构建

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn.net/weixin_39470744/arti ...

  2. 谷歌BERT预训练源码解析(三):训练过程

    目录 前言 源码解析 主函数 自定义模型 遮蔽词预测 下一句预测 规范化数据集 前言 本部分介绍BERT训练过程,BERT模型训练过程是在自己的TPU上进行的,这部分我没做过研究所以不做深入探讨.BE ...

  3. 谷歌BERT预训练源码解析(一):训练数据生成

    目录 预训练源码结构简介 输入输出 源码解析 参数 主函数 创建训练实例 下一句预测&实例生成 随机遮蔽 输出 结果一览 预训练源码结构简介 关于BERT,简单来说,它是一个基于Transfo ...

  4. Gin源码解析和例子——中间件(middleware)

    在<Gin源码解析和例子--路由>一文中,我们已经初识中间件.本文将继续探讨这个技术.(转载请指明出于breaksoftware的csdn博客) Gin的中间件,本质是一个匿名回调函数.这 ...

  5. Colly源码解析——结合例子分析底层实现

    通过<Colly源码解析--框架>分析,我们可以知道Colly执行的主要流程.本文将结合http://go-colly.org上的例子分析一些高级设置的底层实现.(转载请指明出于break ...

  6. libev源码解析——定时器监视器和组织形式

    我们先看下定时器监视器的数据结构.(转载请指明出于breaksoftware的csdn博客) /* invoked after a specific time, repeatable (based o ...

  7. libev源码解析——定时器原理

    本文将回答<libev源码解析--I/O模型>中抛出的两个问题.(转载请指明出于breaksoftware的csdn博客) 对于问题1:为什么backend_poll函数需要指定超时?我们 ...

  8. libev源码解析——I/O模型

    在<libev源码解析--总览>一文中,我们介绍过,libev是一个基于事件的循环库.本文将介绍其和事件及循环之间的关系.(转载请指明出于breaksoftware的csdn博客) 目前i ...

  9. libev源码解析——调度策略

    在<libev源码解析--监视器(watcher)结构和组织形式>中介绍过,监视器分为[2,-2]区间5个等级的优先级.等级为2的监视器最高优,然后依次递减.不区分监视器类型和关联的文件描 ...

最新文章

  1. JSP实现大学生综合素质测评系统
  2. 苏州银行签约神策数据,致力打造科技引领的新时代普惠银行
  3. mysql查看执行计划_如何查看MySQL的执行计划
  4. 牛客题霸 [括号序列] C++题解/答案
  5. HDU 2056 Rectangles
  6. java class文件比较_java class文件查看工具
  7. 短信炸弹jmeter验证方法
  8. python 姓名转拼音
  9. 机器学习模型的超参数优化
  10. Windows创建快捷方式的几种方法你用过哪些?
  11. 实用小技巧 利用Python一秒将全部中文名转为拼音
  12. 社交巨人屏蔽Google搜索 Facebook发展强劲拟上市
  13. androidmmi可以卸载吗_删除Android自带的系统软件注意事项
  14. Modem analyze by T32 tool.
  15. 鼠标滚轮失灵上下乱跳的解决办法
  16. 无Internet访问权限-已解决
  17. 什么是Java SE、Java EE、Java ME?
  18. [MB855]变砖解决
  19. 新闻类客户端代码阅读笔记
  20. 校验码中码距与纠错能力的关系

热门文章

  1. 第 5 章 虚拟机栈
  2. 计算机维护宝典,超齐全的维修宝典之电脑维修实例大全
  3. moba寻路_MOBA代号105:道具收费 加入自动寻路等MMO元素
  4. beautifulsoup_如何使用 Python 和 BeautifulSoup 爬取网站
  5. matlab函数anova,MATLAB进行单因素方差分析-ANOVA
  6. leftjoin多表联合查询_结合mybatis-plus 实现无XML多表联合查询
  7. mysql innodb row_format_INNODB物理行结构(ROW_FORMAT=COMPACT) | 学步园
  8. python 进位_蓝桥杯-Python-高精度加法
  9. admin.php c install,laravel-admin后台的安装
  10. 腾讯云服务器CentOS 7安装MariaDB并用Navicat Premium连接