`

利用redis分布式锁的功能来实现定时器的分布式

阅读更多

        以前为部门内部开发过一个定时器程序,这个定时器很简单,就是配置quartz,来实现定时调用配置的url功能。最近为了防止定时器所在的服务器由于特殊原因挂掉,需要对定时器做多机部署。那么如果按照原来的方式进行部署,就会遇到 在一定的间隔时间内,可能出现多次重复调用的问题。为了解决这个问题,我就借助了redis的分布式锁功能。

        redis分布式锁参考 : http://www.jeffkit.info/2011/07/1000/

        具体原理如下:

        定时器到时间被触发,程序开始先争取一个redis锁。

        如果获得锁,就设置锁的超时时间为到下次定时器触发的时间。

        然后执行定时器任务。后来的定时器也来尝试获得redis锁,当然,这个锁已不能获取了,而且超时时间在未来,所以就放弃这次任务调用。

        定时器到时间再次被触发,然后尝试获得锁,由于锁的超时时间为定时任务的时间间隔,当前时间正好大于或等于超时时间,所以,程序可以顺利的获得锁,并重置超时时间。

        。。。。。。。不断的循环调用,判断

 

        在此之间测试循环间隔时间最小单位为1s最好,如果小于1s的调用,由于使用redis会有10几毫秒的运算耗费,因此不能保证在1s以下的时间间隔比较均匀.

        为了能保证定时触发时,能获得redis锁,可以设置锁的超时时间为间隔时间-10ms。这样就判断超时时间 now > timeoutTime = true。

       补充:

       只要多个服务器时间差别不大,基本不会有重复的问题。唯一担心的就是redis,这个挂了,就全挂了。      

       因此,如果要考虑更全面,需要对redis点单再做集群。就看是否有必要了。

 

       目前定时器的任务都是通过配置文件写好,以后我还要考虑动态更新任务。

 

       目前代码还在整理中,想做一个开源的项目。

 

       踩我的请告诉我,哪里不对。如果大家有更好的方案,请告诉我,我需要改进

4
4
分享到:
评论
19 楼 zxjlwt 2015-11-30  
学习了。
http://surenpi.com
18 楼 ak478288 2014-11-13  
yangjianzhouctgu 写道
怎么样保证每次的定时任务被执行?在获得锁的情况下,没执行怎么办?



定时任务是否被执行,取决于任务的业务逻辑。只要定时能被触发,就能执行任务。除非quartz有bug。那么要确认执行成功的话,只能双方协商应答。对于不正常的应答就需要重试。
17 楼 yangjianzhouctgu 2014-11-11  
怎么样保证每次的定时任务被执行?在获得锁的情况下,没执行怎么办?
16 楼 ak478288 2014-09-28  
f529352479 写道
ak478288 写道
f529352479 写道
我觉得是可行的,不知道楼主现在用的怎么样了


现在升级换方案了,使用zookeeper来搞定单点和并发锁的问题。

现在换成什么方案了,可否共享一下,我也有这样的问题需要解决


redis和zookeeper实现一样的功能,就是分布式锁,请google看看相关资料就可以,我们也是基于这样的方案来实现
15 楼 f529352479 2014-09-28  
ak478288 写道
f529352479 写道
我觉得是可行的,不知道楼主现在用的怎么样了


现在升级换方案了,使用zookeeper来搞定单点和并发锁的问题。

现在换成什么方案了,可否共享一下,我也有这样的问题需要解决
14 楼 ak478288 2014-07-31  
f529352479 写道
这个得保证各服务器的时间严格一致,如果有一台服务器上的时间快了一秒,一秒之差就能让该服务器每次都能提前一秒去抢锁,而其他的服务器抢不到锁的几率大大降低,就起不到负载均衡的作用了!


你说的对,服务器时间有问题,就会产生误差
13 楼 ak478288 2014-07-30  
f529352479 写道
我觉得是可行的,不知道楼主现在用的怎么样了


现在升级换方案了,使用zookeeper来搞定单点和并发锁的问题。
12 楼 f529352479 2014-07-27  
这个得保证各服务器的时间严格一致,如果有一台服务器上的时间快了一秒,一秒之差就能让该服务器每次都能提前一秒去抢锁,而其他的服务器抢不到锁的几率大大降低,就起不到负载均衡的作用了!
11 楼 f529352479 2014-07-25  
我觉得是可行的,不知道楼主现在用的怎么样了
10 楼 skzr.org 2013-07-04  
就是你同事这样的的方案(多了可以用xml配置任务以实现任务重用,类似shell的管道原理)

ak478288 写道
skzr.org 写道
ak478288 写道
rox 写道
这个方案不错,灵活方便。
系统资源牺牲比较小。
原来,考虑定时器,都是时间同步,配置麻烦,而且,收效甚微。


嗯,只要多个服务器时间差别不大,基本不会有重复的问题。唯一担心的就是redis,这个挂了,就全挂了。因此,如果要考虑更全面,需要对redis点单再做集群。就看是否有必要了。

目前定时器的任务都是通过配置文件写好,以后我还要考虑动态更新任务。

目前代码还在整理中,想做一个开源的项目。


我之前做过的使用db做状态,多机后台执行调度,每行表示一个task。
1. 定时器每隔1s扫描一次表,安排本地下次执行时间和任务。
2. 任务开始执行前尝试锁定,检查状态,然后更新状态为下次执行时间(同时做记录的ver),任务开始执行。
3. 锁定失败就放弃执行


能详细点说么,你的方案,我们有同事类似的做过。
1、使用的是数据库事务序列化的级别来锁定
2、获取要执行的数据
3、更新这些数据的状态
4、提交事务
5、然后执行任务

我认为用db会复杂化。希望能看到你的详细方案

9 楼 ak478288 2013-07-04  
skzr.org 写道
ak478288 写道
rox 写道
这个方案不错,灵活方便。
系统资源牺牲比较小。
原来,考虑定时器,都是时间同步,配置麻烦,而且,收效甚微。


嗯,只要多个服务器时间差别不大,基本不会有重复的问题。唯一担心的就是redis,这个挂了,就全挂了。因此,如果要考虑更全面,需要对redis点单再做集群。就看是否有必要了。

目前定时器的任务都是通过配置文件写好,以后我还要考虑动态更新任务。

目前代码还在整理中,想做一个开源的项目。


我之前做过的使用db做状态,多机后台执行调度,每行表示一个task。
1. 定时器每隔1s扫描一次表,安排本地下次执行时间和任务。
2. 任务开始执行前尝试锁定,检查状态,然后更新状态为下次执行时间(同时做记录的ver),任务开始执行。
3. 锁定失败就放弃执行


对于你说的 我很想知道是如何锁定,如何判断锁定失败。
8 楼 ak478288 2013-07-04  
skzr.org 写道
ak478288 写道
rox 写道
这个方案不错,灵活方便。
系统资源牺牲比较小。
原来,考虑定时器,都是时间同步,配置麻烦,而且,收效甚微。


嗯,只要多个服务器时间差别不大,基本不会有重复的问题。唯一担心的就是redis,这个挂了,就全挂了。因此,如果要考虑更全面,需要对redis点单再做集群。就看是否有必要了。

目前定时器的任务都是通过配置文件写好,以后我还要考虑动态更新任务。

目前代码还在整理中,想做一个开源的项目。


我之前做过的使用db做状态,多机后台执行调度,每行表示一个task。
1. 定时器每隔1s扫描一次表,安排本地下次执行时间和任务。
2. 任务开始执行前尝试锁定,检查状态,然后更新状态为下次执行时间(同时做记录的ver),任务开始执行。
3. 锁定失败就放弃执行


能详细点说么,你的方案,我们有同事类似的做过。
1、使用的是数据库事务序列化的级别来锁定
2、获取要执行的数据
3、更新这些数据的状态
4、提交事务
5、然后执行任务

我认为用db会复杂化。希望能看到你的详细方案
7 楼 ak478288 2013-07-04  
skzr.org 写道
ak478288 写道
youjianbo_han_87 写道
为了实现A,引入B,结果又要维护B。为了保证B,可能又要引入C,又要维护C。。。。系统就是这么膨胀的。能否从业务角度优化下呢?


其实是可以从业务角度来考虑的。谁都不想维护那么多依赖。但是为了简化业务系统,就需要这么一个独立的定时器系统来解决所有业务系统都需要的功能。

这么做维护是不可避免的。例如你要用数据库,不够用了,就要加入缓存,以后还要考虑数据库和缓存的集群。都是一样的道理。

重点看是要解决问题的方式和引入的问题,哪个更重要。方案没有完美的,只有均衡选择。


呵呵,这个需要考虑下时间,解耦合是好的,不过如果有跟简单的办法实现,就不用复杂的。


谁说不是呢 ,不过目前我没有看到解决的好办法。
6 楼 skzr.org 2013-07-04  
ak478288 写道
youjianbo_han_87 写道
为了实现A,引入B,结果又要维护B。为了保证B,可能又要引入C,又要维护C。。。。系统就是这么膨胀的。能否从业务角度优化下呢?


其实是可以从业务角度来考虑的。谁都不想维护那么多依赖。但是为了简化业务系统,就需要这么一个独立的定时器系统来解决所有业务系统都需要的功能。

这么做维护是不可避免的。例如你要用数据库,不够用了,就要加入缓存,以后还要考虑数据库和缓存的集群。都是一样的道理。

重点看是要解决问题的方式和引入的问题,哪个更重要。方案没有完美的,只有均衡选择。


呵呵,这个需要考虑下时间,解耦合是好的,不过如果有跟简单的办法实现,就不用复杂的。
5 楼 ak478288 2013-07-04  
youjianbo_han_87 写道
为了实现A,引入B,结果又要维护B。为了保证B,可能又要引入C,又要维护C。。。。系统就是这么膨胀的。能否从业务角度优化下呢?


其实是可以从业务角度来考虑的。谁都不想维护那么多依赖。但是为了简化业务系统,就需要这么一个独立的定时器系统来解决所有业务系统都需要的功能。

这么做维护是不可避免的。例如你要用数据库,不够用了,就要加入缓存,以后还要考虑数据库和缓存的集群。都是一样的道理。

重点看是要解决问题的方式和引入的问题,哪个更重要。方案没有完美的,只有均衡选择。
4 楼 youjianbo_han_87 2013-07-04  
为了实现A,引入B,结果又要维护B。为了保证B,可能又要引入C,又要维护C。。。。系统就是这么膨胀的。能否从业务角度优化下呢?
3 楼 skzr.org 2013-07-04  
ak478288 写道
rox 写道
这个方案不错,灵活方便。
系统资源牺牲比较小。
原来,考虑定时器,都是时间同步,配置麻烦,而且,收效甚微。


嗯,只要多个服务器时间差别不大,基本不会有重复的问题。唯一担心的就是redis,这个挂了,就全挂了。因此,如果要考虑更全面,需要对redis点单再做集群。就看是否有必要了。

目前定时器的任务都是通过配置文件写好,以后我还要考虑动态更新任务。

目前代码还在整理中,想做一个开源的项目。


我之前做过的使用db做状态,多机后台执行调度,每行表示一个task。
1. 定时器每隔1s扫描一次表,安排本地下次执行时间和任务。
2. 任务开始执行前尝试锁定,检查状态,然后更新状态为下次执行时间(同时做记录的ver),任务开始执行。
3. 锁定失败就放弃执行
2 楼 ak478288 2013-07-04  
rox 写道
这个方案不错,灵活方便。
系统资源牺牲比较小。
原来,考虑定时器,都是时间同步,配置麻烦,而且,收效甚微。


嗯,只要多个服务器时间差别不大,基本不会有重复的问题。唯一担心的就是redis,这个挂了,就全挂了。因此,如果要考虑更全面,需要对redis点单再做集群。就看是否有必要了。

目前定时器的任务都是通过配置文件写好,以后我还要考虑动态更新任务。

目前代码还在整理中,想做一个开源的项目。
1 楼 rox 2013-07-04  
这个方案不错,灵活方便。
系统资源牺牲比较小。
原来,考虑定时器,都是时间同步,配置麻烦,而且,收效甚微。

相关推荐

    php redis setnx分布式锁简单原理解析

    总的来说,PHP通过Redis的`setnx`实现的分布式锁是一个基本的解决方案,但在实际应用中,需要考虑更多的边缘情况和优化措施以保证系统的稳定性和正确性。通过不断学习和实践,我们可以更好地理解和应用分布式锁,...

    SpringBoot项目中利用Redis实现系统双活_双机热备

    可以采用分布式定时任务框架,如Quartz或Spring Task,配合分布式锁(如Redis Lock)来确保定时任务在任意一台活动服务器上唯一执行。 6. **数据一致性考虑**:尽管Redis有主从复制,但在特定情况下,如网络延迟或...

    基于redis实现定时任务的方法详解

    在IT行业中,定时任务是许多系统不可或缺的一...它利用了Redis的过期事件通知和分布式锁机制,减少了对系统资源的占用,同时也简化了系统的复杂性。通过合理设计键名和业务逻辑,可以构建出稳定可靠的定时任务系统。

    java面试 分布式 集群 mongodb redis

    - **实现方式**:通过定时器定期发送空数据包,并等待对方响应,以此来判断连接的有效性。 **1.2 心跳检测步骤** 1. **客户端**:定期向服务器发送心跳包,并启动超时计时器。 2. **服务器**:接收到心跳包后,...

    Go-分布式定时任务库distributed-cron

    它充分利用了 Go 的并发特性,提供了一种高效的方式来管理和执行周期性的任务,同时支持在多个节点上进行负载均衡和故障恢复。 在分布式系统中,定时任务库扮演着至关重要的角色,它可以帮助我们执行诸如数据同步、...

    Redis面试专题.pdf

    使用Redis设计分布式锁,一种常见的实现思路是利用SETNX命令(SET if Not eXists)来设置一个键值对,如果返回值为1表示成功上锁,返回0表示已有其他客户端上锁。为了处理死锁的情况,可以为键设置一个过期时间。在...

    Redis面试专题及答案(下).pdf

    使用Redis实现分布式锁,线程可以尝试setnx来获取锁,并设置超时时间防止死锁。 ZooKeeper也可以用于实现分布式锁,其通过在ZooKeeper集群中的节点上创建临时有序节点,然后比较节点序号来决定哪个客户端获取了锁。...

    python基于Tornado实现,系统核心调度,可分布式扩展

    Python基于Tornado实现的系统核心调度能够有效地支持分布式扩展,这是一种高效、轻量级的解决方案,尤其适合处理大量并发连接。Tornado是一个Python Web框架和异步网络库,由FriendFeed团队开发,后来被Facebook收购...

    Swoole 分布式通讯框架 SwooleDistributed.zip

    SwooleDistributed 是swoole分布式系统的实现,他提供了一套基于swoole扩展的分布式通讯框架。结构图:SwooleDistributed 不仅提供了分布式搭建的必要设施,还提供了4大组件帮助你提高编写代码的效率,你既可以部署...

    Redis 概览.pptx

    - **并发锁**:利用`Exists`命令来实现分布式锁机制,确保同一时间只有一个进程能够执行某些关键操作。 - **排行榜**:借助有序集合(Sorted Set)结构来构建实时更新的排行榜。 - **消息队列**:利用`BLpop`与`BRpop`...

    基于Spring Boot的分布式系统中间件集成.zip

    - **定时任务**:整合QuartzJob,实现定时器实时管理。 - **权限管理**:整合Shiro和Spring Security,实现用户权限管理。 - **搜索引擎**:整合ElasticSearch,实现高性能搜索引擎。 - **分布式服务**:整合Dubbo...

    java面试redis篇基本面试必问

    3. **分布式锁**:利用setnx或redlock等机制实现分布式环境下的锁服务。 4. **数据类型丰富**:除了基本的键值对,还支持字符串、哈希、列表、集合和有序集合等多种数据结构。 5. **复制与哨兵**:复制功能用于创建...

    go利用context实现过期单机锁

    此外,考虑到单机锁的限制,如果你需要跨多个节点的锁,可能需要考虑分布式锁的实现,如Redis或etcd提供的解决方案。 通过这种方式,`context`包不仅帮助我们实现了过期单机锁,还确保了良好的并发控制和资源管理。...

    Redis面试专题.docx

    在这个Java实现中,我们利用了`LinkedHashMap`的特性来自动维护最近最少使用的元素。当缓存容量达到上限时,`removeEldestEntry`方法会被调用,如果缓存大小超过设定的容量,则最老的条目将被自动移除。

    2、Redis sentinel原理及实现源码剖析-冯光普1

    Redis Sentinel是Redis的高可用性解决方案,它提供监控、故障检测、自动故障转移以及配置提供者的功能。在冯光普的演讲中,他详细解析了Sentinel的工作原理和源码实现。 Sentinel系统的核心特性包括: 1. **分布式...

    tp5.1+swoole+redis实战直播平台 (学习版)-tp5live.zip

    5. 实现分布式锁,保证并发操作的安全性。 **压缩包文件名称列表:tp5live-master** 这表明压缩包包含的是一个名为"tp5live"的项目仓库的主分支(master)代码。通常,这将包括项目的源代码、配置文件、数据库脚本...

    shell监控redis集群节点异常发送邮件通知管理员.rar

    在“shell监控redis集群节点异常发送邮件通知管理员”这个场景中,我们关注的是如何利用Shell脚本实现这一功能。Shell脚本是一种在Unix/Linux操作系统下编写的一系列命令集合,可以实现自动化执行任务。在这个特定的...

Global site tag (gtag.js) - Google Analytics