`
y806839048
  • 浏览: 1117037 次
  • 性别: Icon_minigender_1
  • 来自: 上海
文章分类
社区版块
存档分类
最新评论

redis进程锁的设计思路

阅读更多

redis进程锁的设计思路:

 

1,持锁的才能自己删

 

2,超时的谁都可以删   lock的value就是过期时间即可(持有超时)

 

3,请求锁的设置求情等待超时时间(请求超时)

 

 

  expireMsecs 锁持有超时,防止线程在入锁以后,无限的执行下去,让锁无法释放 

  timeoutMsecs 锁等待超时,防止线程饥饿,永远没有入锁执行代码的机会 

 

 

 

1.通过setnx尝试设置某个key的值,成功(当前没有这个锁)则返回,成功获得锁(持有者自己删除,后者获取redis锁,返回java获取锁的标志,java可以进行下去)

2.锁已经存在则获取锁的到期时间,和当前时间比较,超时的话,则设置新的值(没删除,但是过期,重新设置,返回标志获取锁,Java可以进行下去)

 

1,2合在一起返回true表示可以获得锁(综合redis锁,和过期,形成Java的进程锁标志(依赖redis共享记录)),不在是单纯的setnx

 

 

工具类中不注入redis模板,在使用的时候注入

 

 

 

 

 

jedisLock—redis分布式锁实现

一、使用分布式锁要满足的几个条件:

  1. 系统是一个分布式系统(关键是分布式,单机的可以使用ReentrantLock或者synchronized代码块来实现)
  2. 共享资源(各个系统访问同一个资源,资源的载体可能是传统关系型数据库或者NoSQL)
  3. 同步访问(即有很多个进程同事访问同一个共享资源。没有同步访问,谁管你资源竞争不竞争)

二、应用的场景例子

  管理后台的部署架构(多台tomcat服务器+redis【多台tomcat服务器访问一台redis】+mysql【多台tomcat服务器访问一台服务器上的mysql】)就满足使用分布式锁的条件。多台服务器要访问redis全局缓存的资源,如果不使用分布式锁就会出现问题。 看如下伪代码:

复制代码
long N=0L;
//N从redis获取值
if(N<5){
N++//N写回redis
}
复制代码

上面的代码主要实现的功能:

  从redis获取值N,对数值N进行边界检查,自加1,然后N写回redis中。 这种应用场景很常见,像秒杀,全局递增ID、IP访问限制等。以IP访问限制来说,恶意攻击者可能发起无限次访问,并发量比较大,分布式环境下对N的边界检查就不可靠,因为从redis读的N可能已经是脏数据。传统的加锁的做法(如java的synchronized和Lock)也没用,因为这是分布式环境,这个同步问题的救火队员也束手无策。在这危急存亡之秋,分布式锁终于有用武之地了。

  分布式锁可以基于很多种方式实现,比如zookeeper、redis...。不管哪种方式,他的基本原理是不变的:用一个状态值表示锁,对锁的占用和释放通过状态值来标识。

   这里主要讲如何用redis实现分布式锁。

三、使用redis的setNX命令实现分布式锁  

1、实现的原理

  Redis为单进程单线程模式,采用队列模式将并发访问变成串行访问,且多客户端对Redis的连接并不存在竞争关系。redis的SETNX命令可以方便的实现分布式锁。

2、基本命令解析

1)setNX(SET if Not eXists)

语法:

SETNX key value

将 key 的值设为 value ,当且仅当 key 不存在。

若给定的 key 已经存在,则 SETNX 不做任何动作。

SETNX 是『SET if Not eXists』(如果不存在,则 SET)的简写

返回值:

  设置成功,返回 1 。
  设置失败,返回 0 。

 例子:

复制代码
redis> EXISTS job                # job 不存在
(integer) 0

redis> SETNX job "programmer"    # job 设置成功
(integer) 1

redis> SETNX job "code-farmer"   # 尝试覆盖 job ,失败
(integer) 0

redis> GET job                   # 没有被覆盖
"programmer"
复制代码

 所以我们使用执行下面的命令

SETNX lock.foo <current Unix time + lock timeout + 1> 
  • 如返回1,则该客户端获得锁,把lock.foo的键值设置为时间值表示该键已被锁定,该客户端最后可以通过DEL lock.foo来释放该锁。

  • 如返回0,表明该锁已被其他客户端取得,这时我们可以先返回或进行重试等对方完成或等待锁超时。

2)getSET

语法:

GETSET key value

  将给定 key 的值设为 value ,并返回 key 的旧值(old value)

  当 key 存在但不是字符串类型时,返回一个错误。

返回值:

  返回给定 key 的旧值。
  当 key 没有旧值时,也即是, key 不存在时,返回 nil 。
3)get
语法:
GET key

 返回值:

  当 key 不存在时,返回 nil ,否则,返回 key 的值。
  如果 key 不是字符串类型,那么返回一个错误

四、解决死锁

  上面的锁定逻辑有一个问题:如果一个持有锁的客户端失败或崩溃了不能释放锁,该怎么解决

我们可以通过锁的键对应的时间戳来判断这种情况是否发生了,如果当前的时间已经大于lock.foo的值,说明该锁已失效,可以被重新使用。 

  发生这种情况时,可不能简单的通过DEL来删除锁,然后再SETNX一次(讲道理,删除锁的操作应该是锁拥有这执行的,这里只需要等它超时即可),当多个客户端检测到锁超时后都会尝试去释放它,这里就可能出现一个竞态条件,让我们模拟一下这个场景: 

复制代码
C0操作超时了,但它还持有着锁,C1和C2读取lock.foo检查时间戳,先后发现超时了。 
C1 发送DEL lock.foo 
C1 发送SETNX lock.foo 并且成功了。 
C2 发送DEL lock.foo 
C2 发送SETNX lock.foo 并且成功了。 
这样一来,C1,C2都拿到了锁!问题大了! 
复制代码

  幸好这种问题是可以避免的,让我们来看看C3这个客户端是怎样做的: 

复制代码
C3发送SETNX lock.foo 想要获得锁,由于C0还持有锁,所以Redis返回给C3一个0 
C3发送GET lock.foo 以检查锁是否超时了,如果没超时,则等待或重试。 
反之,如果已超时,C3通过下面的操作来尝试获得锁: 
GETSET lock.foo <current Unix time + lock timeout + 1> 
通过GETSET,C3拿到的时间戳如果仍然是超时的,那就说明,C3如愿以偿拿到锁了。 
如果在C3之前,有个叫C4的客户端比C3快一步执行了上面的操作,那么C3拿到的时间戳是个未超时的值,这时,C3没有如期获得锁,需要再次等待或重试。留意一下,尽管C3没拿到锁,但它改写了C4设置的锁的超时值,不过这一点非常微小的误差带来的影响可以忽略不计。 
复制代码

  注意:为了让分布式锁的算法更稳键些,持有锁的客户端在解锁之前应该再检查一次自己的锁是否已经超时,再去做DEL操作,因为可能客户端因为某个耗时的操作而挂起,操作完的时候锁因为超时已经被别人获得,这时就不必解锁了。  

五、代码实现

  expireMsecs 锁持有超时,防止线程在入锁以后,无限的执行下去,让锁无法释放 
  timeoutMsecs 锁等待超时,防止线程饥饿,永远没有入锁执行代码的机会 

注意:项目里面需要先搭建好redis的相关配置

复制代码
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.dao.DataAccessException;
import org.springframework.data.redis.connection.RedisConnection;
import org.springframework.data.redis.core.RedisCallback;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.StringRedisSerializer;

/**
 * Redis distributed lock implementation.
 *
 * @author zhengcanrui
 */
public class RedisLock {

    private static Logger logger = LoggerFactory.getLogger(RedisLock.class);

    private RedisTemplate redisTemplate;

    private static final int DEFAULT_ACQUIRY_RESOLUTION_MILLIS = 100;

    /**
     * Lock key path.
     */
    private String lockKey;

    /**
     * 锁超时时间,防止线程在入锁以后,无限的执行等待
     */
    private int expireMsecs = 60 * 1000;

    /**
     * 锁等待时间,防止线程饥饿
     */
    private int timeoutMsecs = 10 * 1000;

    private volatile boolean locked = false;

    /**
     * Detailed constructor with default acquire timeout 10000 msecs and lock expiration of 60000 msecs.
     *
     * @param lockKey lock key (ex. account:1, ...)
     */
    public RedisLock(RedisTemplate redisTemplate, String lockKey) {
        this.redisTemplate = redisTemplate;
        this.lockKey = lockKey + "_lock";
    }

    /**
     * Detailed constructor with default lock expiration of 60000 msecs.
     *
     */
    public RedisLock(RedisTemplate redisTemplate, String lockKey, int timeoutMsecs) {
        this(redisTemplate, lockKey);
        this.timeoutMsecs = timeoutMsecs;
    }

    /**
     * Detailed constructor.
     *
     */
    public RedisLock(RedisTemplate redisTemplate, String lockKey, int timeoutMsecs, int expireMsecs) {
        this(redisTemplate, lockKey, timeoutMsecs);
        this.expireMsecs = expireMsecs;
    }

    /**
     * @return lock key
     */
    public String getLockKey() {
        return lockKey;
    }

    private String get(final String key) {
        Object obj = null;
        try {
            obj = redisTemplate.execute(new RedisCallback<Object>() {
                @Override
                public Object doInRedis(RedisConnection connection) throws DataAccessException {
                    StringRedisSerializer serializer = new StringRedisSerializer();
                    byte[] data = connection.get(serializer.serialize(key));
                    connection.close();
                    if (data == null) {
                        return null;
                    }
                    return serializer.deserialize(data);
                }
            });
        } catch (Exception e) {
            logger.error("get redis error, key : {}", key);
        }
        return obj != null ? obj.toString() : null;
    }

    private boolean setNX(final String key, final String value) {
        Object obj = null;
        try {
            obj = redisTemplate.execute(new RedisCallback<Object>() {
                @Override
                public Object doInRedis(RedisConnection connection) throws DataAccessException {
                    StringRedisSerializer serializer = new StringRedisSerializer();
                    Boolean success = connection.setNX(serializer.serialize(key), serializer.serialize(value));
                    connection.close();
                    return success;
                }
            });
        } catch (Exception e) {
            logger.error("setNX redis error, key : {}", key);
        }
        return obj != null ? (Boolean) obj : false;
    }

    private String getSet(final String key, final String value) {
        Object obj = null;
        try {
            obj = redisTemplate.execute(new RedisCallback<Object>() {
                @Override
                public Object doInRedis(RedisConnection connection) throws DataAccessException {
                    StringRedisSerializer serializer = new StringRedisSerializer();
                    byte[] ret = connection.getSet(serializer.serialize(key), serializer.serialize(value));
                    connection.close();
                    return serializer.deserialize(ret);
                }
            });
        } catch (Exception e) {
            logger.error("setNX redis error, key : {}", key);
        }
        return obj != null ? (String) obj : null;
    }

    /**
     * 获得 lock.
     * 实现思路: 主要是使用了redis 的setnx命令,缓存了锁.
     * reids缓存的key是锁的key,所有的共享, value是锁的到期时间(注意:这里把过期时间放在value了,没有时间上设置其超时时间)
     * 执行过程:
     * 1.通过setnx尝试设置某个key的值,成功(当前没有这个锁)则返回,成功获得锁
     * 2.锁已经存在则获取锁的到期时间,和当前时间比较,超时的话,则设置新的值
     *
     * @return true if lock is acquired, false acquire timeouted
     * @throws InterruptedException in case of thread interruption
     */
    public synchronized boolean lock() throws InterruptedException {
        int timeout = timeoutMsecs;
        while (timeout >= 0) {
            long expires = System.currentTimeMillis() + expireMsecs + 1;
            String expiresStr = String.valueOf(expires); //锁到期时间
            if (this.setNX(lockKey, expiresStr)) {
                // lock acquired
                locked = true;
                return true;
            }

            String currentValueStr = this.get(lockKey); //redis里的时间
            if (currentValueStr != null && Long.parseLong(currentValueStr) < System.currentTimeMillis()) {
                //判断是否为空,不为空的情况下,如果被其他线程设置了值,则第二个条件判断是过不去的
                // lock is expired

                String oldValueStr = this.getSet(lockKey, expiresStr);
                //获取上一个锁到期时间,并设置现在的锁到期时间,
                //只有一个线程才能获取上一个线上的设置时间,因为jedis.getSet是同步的
                if (oldValueStr != null && oldValueStr.equals(currentValueStr)) {
                    //防止误删(覆盖,因为key是相同的)了他人的锁——这里达不到效果,这里值会被覆盖,但是因为什么相差了很少的时间,所以可以接受

                    //[分布式的情况下]:如过这个时候,多个线程恰好都到了这里,但是只有一个线程的设置值和当前值相同,他才有权利获取锁
                    // lock acquired
                    locked = true;
                    return true;
                }
            }
            timeout -= DEFAULT_ACQUIRY_RESOLUTION_MILLIS;

            /*
                延迟100 毫秒,  这里使用随机时间可能会好一点,可以防止饥饿进程的出现,即,当同时到达多个进程,
                只会有一个进程获得锁,其他的都用同样的频率进行尝试,后面有来了一些进行,也以同样的频率申请锁,这将可能导致前面来的锁得不到满足.
                使用随机的等待时间可以一定程度上保证公平性
             */
            Thread.sleep(DEFAULT_ACQUIRY_RESOLUTION_MILLIS);

        }
        return false;
    }


    /**
     * Acqurired lock release.
     */
    public synchronized void unlock() {
        if (locked) {
            redisTemplate.delete(lockKey);
            locked = false;
        }
    }

}
复制代码

 调用:

复制代码
 RedisLock lock = new RedisLock(redisTemplate, key, 10000, 20000);
 try {
            if(lock.lock()) {
                   //需要加锁的代码
                }
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            //为了让分布式锁的算法更稳键些,持有锁的客户端在解锁之前应该再检查一次自己的锁是否已经超时,再去做DEL操作,因为可能客户端因为某个耗时的操作而挂起,
            //操作完的时候锁因为超时已经被别人获得,这时就不必解锁了。 ————这里没有做
            lock.unlock();
        }
复制代码

六、一些问题

1、为什么不直接使用expire设置超时时间,而将时间的毫秒数其作为value放在redis中?

如下面的方式,把超时的交给redis处理:

lock(key, expireSec){
isSuccess = setnx key
if (isSuccess)
expire key expireSec
}

  这种方式貌似没什么问题,但是假如在setnx后,redis崩溃了,expire就没有执行,结果就是死锁了。锁永远不会超时。

 2、为什么前面的锁已经超时了,还要用getSet去设置新的时间戳的时间获取旧的值,然后和外面的判断超时时间的时间戳比较呢?

  因为是分布式的环境下,可以在前一个锁失效的时候,有两个进程进入到锁超时的判断。如:

C0超时了,还持有锁,C1/C2同时请求进入了方法里面

C1/C2获取到了C0的超时时间

C1使用getSet方法

C2也执行了getSet方法

假如我们不加 oldValueStr.equals(currentValueStr) 的判断,将会C1/C2都将获得锁,加了之后,能保证C1和C2只能一个能获得锁,一个只能继续等待。

注意:这里可能导致超时时间不是其原本的超时时间,C1的超时时间可能被C2覆盖了,但是他们相差的毫秒及其小,这里忽略了。

 

 参考:

http://www.cnblogs.com/0201zcr/p/5942748.html

分享到:
评论

相关推荐

    Redis面试专题1

    在分布式锁的设计上,Redis 提供了一种基于 SETNX 和 GETSET 命令的实现思路,确保锁的公平性和防止死锁。而 ZooKeeper 作为另一种分布式协调服务,可以通过创建瞬时有序节点来实现锁服务,虽然实现相对简单,但性能...

    如何实现靠谱的分布式锁技术.docx

    SharkLock作为一种新型的分布式锁解决方案,通过其独特的设计思路,在一定程度上解决了传统方案中存在的问题,提供了一个更加稳定和可靠的分布式锁服务。在实际应用中,开发者应当充分理解每种方案的特点,结合实际...

    分布式锁简单实现

    为了深入理解这个项目,你需要查看源代码,分析其设计思路和实现细节。这将帮助你学习到如何在实际场景中应用分布式锁,并为解决性能问题提供思路。此外,可以对比其他成熟的分布式锁实现,如Redisson、Zookeeper等...

    zookeeper经典应用场景

    但是,基于ZooKeeper设计思路存在羊群效应问题,可以使用临时有序znode来表示获取锁的请求,创建最小后缀数字znode的用户成功拿到锁。 Curator可重入分布式锁 在实际的开发中,如果需要使用到分布式锁,不建议去...

    95丨项目实战二:设计实现一个通用的接口幂等框架(实现)1

    实现思路是调用方根据预设算法生成幂等号,框架负责幂等号的存储和判重逻辑,使用Redis作为数据存储并利用其原子性操作避免并发问题。 9. **代码审查与重构**:在实现过程中,需要进行代码审查,发现潜在问题,并...

    Go-系统V消息队列的一个人Go封装

    5. **接口设计**:设计简洁易用的Go接口,如`NewMessageQueue`用于创建队列,`SendMessage`和`ReceiveMessage`用于发送和接收消息,使得Go开发者可以像使用其他Go库一样使用这个封装。 6. **示例和文档**:提供清晰...

    java面试评价表

    - **单线程并发处理**:探究Redis单进程单线程架构如何支撑高并发访问。 - **Redis分布式锁**:介绍如何利用Redis实现分布式锁,以及操作的原子性保障机制。 #### 九、项目经验 - **项目架构与技术选型**:分享...

    基于GO语言大型企业级电商秒杀系统实战教程

    设计思路将请求拦截在系统上游,降低下流压力;秒杀系统特点就是并发量极大,但实际秒杀成功的请求数量确很少,所以如果不在前端拦截可能造成数据库读写锁冲突,甚至导致死锁,最终请求超时,甚至导致系统崩溃充分...

    java2018面试宝典

    - Redis部分涉及Redis的设计与实现,以及Redis实战应用。 - 正则表达式的使用也是面试中经常考察的内容。 - 编码实践部分可以参考《重构:改善既有代码的设计》和《编写可读代码的艺术》。 - 代码风格规范建议参考...

    程序员面试宝典电子版

    3. UML建模:类图、序列图、用例图等,用于表达系统设计思路。 四、编程实战 1. 代码质量:代码规范、异常处理、错误日志、性能优化、单元测试等,体现专业素养。 2. 工具使用:版本控制工具Git,IDE(如IntelliJ ...

    即时通讯软件MyICQ的源代码

    源代码是这类软件的核心,它揭示了软件内部的工作原理和设计思路,对于开发者来说,是一份宝贵的教育资源和研究材料。 MyICQ作为一个开源的即时通讯软件,它的源代码为学习和理解即时通讯系统的架构提供了难得的...

    群星端+源码+数据库+线程.7z

    源码是程序的原始形态,通过阅读和理解源码,开发者可以学习到软件的设计思路、算法实现以及最佳实践。在游戏服务器端的源码中,可能会看到如状态机设计、网络协议解析、游戏逻辑控制、数据库操作等模块。对于初学者...

    面试题问题合集.docx

    ZooKeeper 可以用来实现分布式锁,基本思路是在ZooKeeper上创建临时顺序节点,获取最小编号的节点即获得锁。具体实现可以参考`InterProcessMutex`类。 ### Spring Boot 的使用 #### 添加第三方Jar包 在Spring ...

    最新Python3.5零基础+高级+完整项目(28周全)培训视频学习资料

    进程锁与进程池详解 协程 协程Gevent 协程之爬虫 协程之Socket IO多路复用 IO模式 Select解析Socket通信 作业 第11周 鸡汤 消息队列介绍 RabbitMQ基本示例 RabbitMQ消息分发轮询 RabbitMQ消息持久化 RabbitMQ ...

    1000道 互联网Java工程师面试题 485页 .pdf

    MySQL是常用的数据库,面试中会问到SQL语句优化、索引原理、事务隔离级别、锁机制等。另外,NoSQL数据库如Redis、MongoDB的使用场景和特性也是考察点。Redis的内存管理、持久化策略,以及与Memcached的区别是常见...

    重要基础知识.docx

    13. **系统设计**:涵盖架构设计、负载均衡、容错机制、扩展性等,涉及高并发、大数据处理、分布式服务等挑战。 14. **Servlet的生命周期和工作原理**:Servlet的生命周期包括加载、初始化、服务、销毁四个阶段,...

    技术之瞳 阿里巴巴面试笔试 高清pdf

    5. **分布式系统**:分布式计算、分布式存储、分布式锁、CAP定理、一致性哈希等分布式系统的设计与实现。 6. **算法与数据结构**:面试中经常会遇到实际的编程题目,要求现场编写算法,如查找、排序、图论问题等,...

    Sistemas-Distribuidos

    通过具体的案例,如构建分布式爬虫、分布式搜索引擎或实时流处理系统,深入理解分布式系统的设计思路和Python在其中的应用。 总结,分布式系统是现代互联网技术的重要支柱,Python作为强大的工具,让开发者能够高效...

    python入门到高级全栈工程师培训 第3期 附课件代码

    11 FTP之创建文件夹及MD5校验思路 第33章 01 操作系统历史 02 进程的概念 03 线程的概念 04 线程的调用以及join方法 05 setDaemon方法和继承式调用.baiduyun.downloading 05 setDaemon方法和继承式调用 第34章 ...

Global site tag (gtag.js) - Google Analytics