分布式锁实践指南:Redis篇

目前越来越多的应用使用负载均衡,以往传统单体应用单机部署的情况下使用的JAVA并发处理资源竞争方式(J.U.C或synchronized等)在集群部署中已经无法保证资源的安全访问。


为什么需要分布式锁

需要考虑以下情况——

  • 只允许一个客户端操作共享资源:这种情况下,对共享资源的操作一般是非幂等性操作。在这种情况下,如果出现多个客户端操作共享资源,就可能意味着数据不一致,数据丢失。
  • 允许多个客户端操作共享资源:对共享资源的操作一定是幂等性操作,无论你操作多少次都不会出现不同结果。在这里使用锁,无外乎就是为了避免重复操作共享资源从而提高效率。

为了解决分布式应用中对资源的安全访问于是便有了分布式锁。

实现分布式锁一般基于Zookeeper或者Redis,前者可靠性高而后者效率高

如果并发量不大追求可靠性选择Zookeeper,反之选择Redis。


Redis实现分布式锁

分布式锁一大特点就是排他性,临界条件下仅有获得锁的调用者才能访问资源。

而Redis是基于内存设计K-V数据库且单线程执行命令。

因此使用Redis作为分布式锁的中间件满足两个重要条件——

  • 基于内部:加锁/解锁效率高
  • 单线程:请求先后顺序执行没有并发冲突,所以任意时刻只有一个调用者能成功获取锁。

而且 Redis支持集群部署(sentinel, cluster)保障了可用性。


加锁

使用SETNX()便可以完成加锁操作。SETNX表示如果Key不存在则写入,如果存在则什么都不做。

setnx key value

这样便只允许一个调用者可以访问。等等似乎少了什么?没错我们没有设置KEY的过期时间,如果此时调用者宕机这把锁就无法释放了。

我们使用EXPIRE加上过期时间,默认单位为秒。

EXPIRE key seconds

但是这样做就完了吗?SETNXEXPIRE是两个独立的操作,即加锁操作不具备原子性。假如加锁调用成功,但是设置过期时间的时候调用服务宕机,依然存在锁无法正常释放的问题。

于是我们还需要把这两条命令合为一条命令。

SET key value [expiration EX seconds|PX milliseconds] [NX|XX]
  • expiration:过期时间,EX表示秒,PX表示毫秒
  • NX:表示如果不存在则写入(显然我们需要这条语句)
  • XX:表示如果存在则写入

目前终于算是完成了原子加锁命令并且锁存在过期时间即使调用者宕机到期之后也会自动释放。


解锁

使用DEL便可以删除锁。

DEL key [key ...]

不过事情真的有这么简单吗?当然不会(套路啊)

  • 假设调用者A加锁成功并且设置锁超时时间为30秒。但是因为某些原因导致调用者A处理业务逻辑所花费的时间超过了30秒。
  • 锁超时自动释放之后调用者B刚好加锁成功,这时调用者A使用DEL语句释放了调用者B的锁。
  • 因为锁被调用者A释放导致后续的调用者可以参与锁竞争,例如调用者C获取到锁。
  • 从而发送在同一个时间内调用者B与调用者C同时运行业务逻辑从而破坏了资源的安全访问。

别绕晕了画张图更直观的感受下——

从图中可以发现A、B有并行,B、C有并行不符合我们的要求。

那么怎样才能让调用者只能释放自己占用的锁呢?这个时候K-V中的V就发挥作用了。如果我们的Value都是唯一(例如:UUID)的并且调用者释放锁需要验证Value便可以避免释放其他调用者占用的锁资源。

SET key uuid EX times NX

那么在释放锁的时候就跟需要分三步——

  1. 根据KEY获取锁
  2. 对比VALUE与预期值是否一致
  3. 如果一致则释放(删除)锁

可是这三个步骤是独立非原子操作,如果在2、3步骤之间锁因为超时自动释放且在高并发情况下别其他调用者加锁成功,在执行步骤3时则仍然存在删除其他调用者的锁。

因此我们仍然需要原子操作。遗憾的是Redis的命令并不提供该操作。

别慌Redis虽然没有提供,但是Redis支持LUA脚本,LUA脚本可以帮助原子性的完成这一系列复合操作。

以下为Redis官方提供脚本——

if redis.call('get', KEYS[1]) == ARGV[1] 
then 
  return redis.call('del', KEYS[1]) 
else 
  return 0 
end

这个脚本即使你不会LUA大概也猜得到是什么意思吧。

  1. 传入的键名参数调用get操作获取到的值与附加参数比较
  2. 如果值一致则调用del操作(删除操作成功返回1)
  3. 如果值不一致则返回0

执行脚本如下——

eval "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end" 1 lock value

现在释放锁的原子操作也满足了,让我们看看代码实践。


代码示例

笔者这里使用springboot来演示分布式锁。

因为分布式锁的实现可以是多种方式建议抽象一个接口,例如DistributedLock

如果使用Redis作为分布式锁中间件则创建RedisDistributedLock

如果使用Zookeeper作为分布式锁的中间件则创建ZookeeperDistributedLock

可以根据@ConditionalOnBean以及spring优先级选择加载那种分布式锁。

  • 分布式锁配置

    spring:
      redis:
        lettuce:
          pool:
            max-active: 100
            max-idle: 50
            min-idle: 30
            max-wait: 2000ms
        # 哨兵模式
        sentinel:
          master: mymaster
          nodes: ${ARGS_REDIS_NODES}
        # 集群模式
        # cluster:
          # nodes: ${ARGS_REDIS_NODES}
        password: ${ARGS_REDIS_PASSWORD}
    
    
    # 分布式配置
    distributed:
      lock:
        expire: 30000
        # connection-timeout-ms: 15000
        # session-timeout-ms: 60000
    
  • 分布式锁接口

    public interface DistributedLock {
    
        /**
         * 获取锁
         *
         * @param key
         * @param value
         * @return
         */
        public boolean tryLock(String key, String value);
    
        /**
         * 释放锁
         *
         * @param key
         * @param value
         * @return
         */
        public boolean releaseLock(String key, String value);
    
    }
    
  • 分布式锁实现

    /**
     * 分布式锁:使用Redis实现
     */
    @AutoConfigureAfter({RedisAutoConfiguration.class})
    @ConditionalOnBean(RedisAutoConfiguration.class)
    @Service
    @Slf4j
    public class RedisDistributedLock implements DistributedLock {
    
        private final static String OK = "OK";
    
        private static final String UNLOCK_LUA;
    
        /**
         * Lua脚本来释放锁
         */
        static {
            StringBuilder sb = new StringBuilder();
            sb.append("if redis.call('get', KEYS[1]) == ARGV[1] ");
            sb.append("then ");
            sb.append("    return redis.call('del', KEYS[1]) ");
            sb.append("else ");
            sb.append("    return 0 ");
            sb.append("end");
            UNLOCK_LUA = sb.toString();
        }
    
        /**
         * 缺省超时时间
         */
        @Value("${distributed.lock.expire}")
        private long expire;
    
        @Resource
        private StringRedisTemplate stringRedisTemplate;
    
        @Override
        public boolean tryLock(String key, String value) {
            return tryLock(key, value, TimeUnit.MILLISECONDS, expire);
        }
    
        @Override
        public boolean releaseLock(String key, String value) {
            try {
                return stringRedisTemplate.execute(
                  (RedisCallback<Boolean>) connection -> connection.eval(
                        UNLOCK_LUA.getBytes(), ReturnType.BOOLEAN, 1,
                        key.getBytes(Charset.forName("UTF-8")),
                        value.getBytes(Charset.forName("UTF-8"))));
            } catch (Exception ex) {
                log.error("释放redis锁失败", ex);
            }
            return false;
    
        }
    
        /**
         * 获取锁
         *
         * @param key
         * @param value
         * @param timeUnit
         * @param time
         * @return
         */
        private boolean tryLock(String key, String value, 
                                TimeUnit timeUnit, long time) {
            try {
                return stringRedisTemplate.execute(
                  (RedisCallback<Boolean>) connection -> connection.set(
                        key.getBytes(Charset.forName("UTF-8")),
                        value.getBytes(Charset.forName("UTF-8")),
                        Expiration.milliseconds(timeUnit.toMillis(time)),
                        RedisStringCommands.SetOption.SET_IF_ABSENT)
                );
            } catch (Exception ex) {
                log.error("获取redis锁失败", ex);
            }
            return false;
        }
    
    }
    
  • 调用示例

    boolean lock = distributedLock.tryLock(LOCK_NAME, lockValue);
    try {
      if (!lock) {
        log.info("已存在分布式锁:[{}]", LOCK_NAME);
        return;
      }
    } finally {
      if (lock) {
        distributedLock.releaseLock(LOCK_NAME, lockValue);
      }
    }
    

以上就是对分布式锁的代码演示。

分布式锁的疑问

Redis实现分布式锁还有哪些问题呢?

  • 自动续期

    当业务逻辑处理时间超过锁的过期时间需要有监控线程在过期之前进续期。你可以将其称之为Monitor或者Watchdog

  • 可重入

    分布式锁能够支持一个线程对资源的重复加锁吗?

  • 读写锁

    所谓读写锁即:读读不互斥,读写互斥,写写互斥。例如:ReentrantReadWriteLock。能否想使用J.U.C包下面的类一样使用分布式锁呢?

  • 集群

    Redis集群环境是一个复杂的逻辑结构,节点的上线下线、主从复制、从节点的提升导致的数据丢失考虑了吗?

是不是感觉头都大了。

好在Redis官方给我们指了一条明路。这就是接下来笔者要介绍的内容了。


Redisson介绍

Redisson正式笔者要介绍的Redis分布式锁终极解决方案。什么你没听说过?没关系跟着笔者进入新世界的大门。


什么是Redisson?

Redisson是一个在Redis的基础上实现的Java驻内存数据网格(In-Memory Data Grid)。它不仅提供了一系列的分布式的Java常用对象,还提供了许多分布式服务。其中包括(BitSet, Set, Multimap, SortedSet, Map, List, Queue, BlockingQueue, Deque, BlockingDeque, Semaphore, Lock, AtomicLong, CountDownLatch, Publish / Subscribe, Bloom filter, Remote service, Spring cache, Executor service, Live Object service, Scheduler service) Redisson提供了使用Redis的最简单和最便捷的方法。Redisson的宗旨是促进使用者对Redis的关注分离(Separation of Concern),从而让使用者能够将精力更集中地放在处理业务逻辑上。

关于Redisson项目的详细介绍可以在官方网站找到。每个Redis服务实例都能管理多达1TB的内存。

能够完美的在云计算环境里使用,并且支持AWS ElastiCache主备版AWS ElastiCache集群版Azure Redis Cache阿里云(Aliyun)的云数据库Redis版

一言以蔽之可以像使用本地J.U.C一样使用Redis分布式锁。


Redisson示例

  • 新增依赖支持

    <dependency>
      <groupId>org.redisson</groupId>
      <artifactId>redisson</artifactId>
      <version>3.13.2</version>
    </dependency>
    
  • YAML文件配置

    redisson:
      client:
        sentinel-address:
          - redis://${redis.senntinel.node1}
          - redis://${redis.senntinel.node2}
          - redis://${redis.senntinel.node3}
        master-name: mymaster
        database: 0
        password: ${redis.password}
    
  • 新增RedissonConfig配置类

    @Configuration
    @ConfigurationProperties(prefix = "redisson.client")
    @Getter
    @Setter
    public class RedissonConfig {
      
      private String masterName;
      
      private int database;
      
      private String password;
      
      private String[] sentinelAddress;
      
      @Bean(destroyMethod = "shutdown")
      public RedissonClient redissonClient() {
        Config config = new Config();
        config.useSentinelServers()
          .setMasterName(masterName)
          .setPassword(password)
          .setDatabase(database)
          .setCheckSentinelsList(false)
          .addSentinelAddress(sentinelAddress);
        return Redisson.create(config);
      }
    
    }
    
  • 测试方法

    @GetMapping(value = "/api/lock")
    public String testRedisson() throws InterruptedException {
      // 锁对象
      RLock rLock = redissonClient.getLock("lock");
      // rLock.tryLock();
      // 阻塞式获取锁
      rLock.lock();
      try {
        log.info("the lock for {}-{}", Thread.currentThread().getName(), Thread.currentThread().getId());
        // 超过默认的30秒锁过期时间
        Thread.sleep(1000 * 120);
      } finally {
        rLock.unlock();
        log.info("release the lock for {}-{}", Thread.currentThread().getName(), 
                 Thread.currentThread().getId());
      }
      return "";
    }
    

    验证结果如下所示——

    EDB3D857-1A1C-4A6C-9915-C350705E40A2.jpg

    由此可以Redisson帮我们自动续期。抓取一下Redis的请求信息看看——

    "EVAL" "if (redis.call('exists', KEYS[1]) == 0) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; return redis.call('pttl', KEYS[1]);" "1" "lock" "30000" "edabe2d0-b3d5-4526-b69a-952f1fc994cc:97"
    
    "EVAL" "if (redis.call('exists', KEYS[1]) == 0) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; return redis.call('pttl', KEYS[1]);" "1" "lock" "30000" "edabe2d0-b3d5-4526-b69a-952f1fc994cc:91"
    


Redisson代码

观察代码发现使用LUA脚本续期,Redisson中大量使用LUA脚本保证其命令具有原子性。

1594739476.909029 [0 101.88.95.75:25242] "EVAL" "if (redis.call('exists', KEYS[1]) == 0) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; return redis.call('pttl', KEYS[1]);" "1" "lock" "30000" "edabe2d0-b3d5-4526-b69a-952f1fc994cc:97"
1594739476.909135 [0 lua] "exists" "lock"
1594739476.909158 [0 lua] "hincrby" "lock" "edabe2d0-b3d5-4526-b69a-952f1fc994cc:97" "1"
1594739476.909176 [0 lua] "pexpire" "lock" "30000"
1594739476.909201 [0 101.88.95.75:25242] "WAIT" "2" "1000"
1594739477.166176 [0 10.10.10.101:36452] "PING"
1594739477.452893 [0 10.10.10.101:36468] "PING"
1594739477.619219 [0 10.10.10.101:36468] "PUBLISH" "__sentinel__:hello" "172.17.0.3,26380,2d8a2463c5242b26b0fc58a07b444ed28fca45c6,0,mymaster,10.10.10.101,6379,0"
1594739477.766886 [0 10.10.10.101:36484] "PING"
1594739477.870853 [0 10.10.10.101:36484] "PUBLISH" "__sentinel__:hello" "172.17.0.3,26381,037ccf65d8178a1aaac14bbb40fc28b2cebd4eac,0,mymaster,10.10.10.101,6379,0"
1594739478.017627 [0 101.88.95.75:25247] "EVAL" "if (redis.call('exists', KEYS[1]) == 0) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; return redis.call('pttl', KEYS[1]);" "1" "lock" "30000" "edabe2d0-b3d5-4526-b69a-952f1fc994cc:91"
1594739478.017727 [0 lua] "exists" "lock"
1594739478.017738 [0 lua] "hexists" "lock" "edabe2d0-b3d5-4526-b69a-952f1fc994cc:91"
1594739478.017747 [0 lua] "pttl" "lock"
1594739478.017758 [0 101.88.95.75:25247] "WAIT" "2" "1000"
1594739478.047182 [0 101.88.95.75:25263] "SUBSCRIBE" "redisson_lock__channel:{lock}"
1594739478.065068 [0 101.88.95.75:25251] "EVAL" "if (redis.call('exists', KEYS[1]) == 0) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then redis.call('hincrby', KEYS[1], ARGV[2], 1); redis.call('pexpire', KEYS[1], ARGV[1]); return nil; end; return redis.call('pttl', KEYS[1]);" "1" "lock" "30000" "edabe2d0-b3d5-4526-b69a-952f1fc994cc:91"
1594739478.065226 [0 lua] "exists" "lock"
1594739478.065241 [0 lua] "hexists" "lock" "edabe2d0-b3d5-4526-b69a-952f1fc994cc:91"
1594739478.065256 [0 lua] "pttl" "lock"
1594739478.065270 [0 101.88.95.75:25251] "WAIT" "2" "1000"

MONITOR日志中也证实了Redisson操作使用了大量的LUA脚本。

Redisson的锁集成了J.U.C下的Lock接口,提供分布式的重入锁读写锁等等,读者完全可以放心大胆的使用Redisson实现分布式锁。

Redisson的时候场景非常多,由于篇幅现在这里笔者就点到为止。建议去Redisson官网去进一步学习。


尾声

一个合格的Redis分布式锁考虑到多少问题。在实际使用中还需要考虑到KEY的设计以及锁的粒度问题。粒度越低影响越少,只锁定访问共享数据的代码尽可能的降低锁的粒度。

Redis分布式锁就介绍到这里,看到这里的你是否有所收获呢?

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容

  • 来自公众号:非科班的科班作者:黎杜 前言 标题使用最近异常火热的微信拍一拍的方式命名,最近拍一拍的玩法被各位网友玩...
    码农小光阅读 1,136评论 0 22
  • 来自公众号:非科班的科班作者:黎杜 前言 标题使用最近异常火热的微信拍一拍的方式命名,最近拍一拍的玩法被各位网友玩...
    夜空_2cd3阅读 387评论 0 4
  • 分布式锁 什么是锁?使用锁的目的是为了控制程序的执行顺序,防止共享资源被多个线程同时访问。为了实现多个线程在一个时...
    缘起缘散_f1a7阅读 222评论 0 0
  • 久违的晴天,家长会。 家长大会开好到教室时,离放学已经没多少时间了。班主任说已经安排了三个家长分享经验。 放学铃声...
    飘雪儿5阅读 7,519评论 16 22
  • 创业是很多人的梦想,多少人为了理想和不甘选择了创业来实现自我价值,我就是其中一个。 创业后,我由女人变成了超人,什...
    亦宝宝阅读 1,805评论 4 1