zoukankan      html  css  js  c++  java
  • redis分布式锁的实现

    一、引题

      工作中遇到了很多高并发的问题,例如奖品发放问题,有可能同一个奖品发送到了多个用户身上,还有可能是多个用户抢占一张票的问题。

    二、处理方法

      (1)采用乐观锁的方式来解决问题,无论什么并发,在数据库层面,所有的请求都是线性的,我们可以采用乐观锁的方式来解决。

      (2)采用redis的分布锁来实现,这样效率会非常的高,减轻数据库的压力。

    三、redis分布式锁的具体实现方式

      1、我所采用的实现方式是:

     1     private static final String LOCK_SUCCESS = "OK";
     2     private static final String SET_IF_NOT_EXIST = "NX";
     3     private static final String SET_WITH_EXPIRE_TIME = "PX";
     4     private static final Long RELEASE_SUCCESS = 1L; 
     5 
     6     /**
     7      * 尝试获取分布式锁
     8      * @param lockKey 锁
     9      * @param requestId 请求标识
    10      * @param expireTime 超期时间
    11      * @return 是否获取成功
    12      */
    13     public Boolean tryGetDistributedLock(String lockKey, String requestId, int expireTime) {
    14         Jedis jedis = this.jedisPool.getResource();
    15         String result = jedis.set(lockKey, requestId, SET_IF_NOT_EXIST, SET_WITH_EXPIRE_TIME, expireTime);
    16         if (LOCK_SUCCESS.equals(result)) {
    17             return true;
    18         }
    19         return false;
    20 
    21     }
    22 
    23      /**
    24      * 释放分布式锁
    25      * @param lockKey 锁
    26      * @param requestId 请求标识
    27      * @return 是否释放成功
    28      */
    29     public Boolean releaseDistributedLock(String lockKey, String requestId) {
    30         Jedis jedis = this.jedisPool.getResource();
    31         String script = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";
    32         Object result = jedis.eval(script, Collections.singletonList(lockKey), Collections.singletonList(requestId));
    33         if (RELEASE_SUCCESS.equals(result)) {
    34             return true;
    35         }
    36         return false;
    37 
    38     }

        以上是加锁和解锁的方式。

      2、具体的应用 

    try{
    String requestId = UUID.randomUUID().toString();
    Boolean flag = tryGetDistributedLock(lock,requestId,1000);
    int n = 0;
    while(!flag){
          //如果没有获取锁,可以尝试下一个lock,如果都没有,则尝试 n 次,退出
         ...
         if(n++>5){ throw new Exception("尝试获取锁失败");}  
         ...  
    }
    if(!flag){
    throw new Exception("尝试获取锁失败");
    } }
    catch(){ }finally{ releaseDistributedLock(lock,requestId); }

    参考url:https://www.cnblogs.com/linjiqin/p/8003838.html

  • 相关阅读:
    eclipse 直接向cloudfoundry部署应用
    jenkins slave节点服务 之 标签
    cloud foundry 中 url map/unmap
    公司Oracle生产库某用户中毒【AfterConnect.sql】
    怎样在 Linux 上查看某个端口的相关信息?
    怎样修改 VS Code 主题?
    怎样安装并编译TypeScript?
    怎样坚持写博客?
    第一个shell脚本
    python处理excel之读:xlrd模块
  • 原文地址:https://www.cnblogs.com/lixiaochao/p/9219380.html
Copyright © 2011-2022 走看看