侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

Redis实现分布式锁的几种方法总结

Redis  /  管理员 发布于 5年前   209

Redis实现分布式锁的几种方法总结

分布式锁是控制分布式系统之间同步访问共享资源的一种方式。在分布式系统中,常常需要协调他们的动作。如果不同的系统或是同一个系统的不同主机之间共享了一个或一组资源,那么访问这些资源的时候,往往需要互斥来防止彼此干扰来保证一致性,在这种情况下,便需要使用到分布式锁。

我们来假设一个最简单的秒杀场景:数据库里有一张表,column分别是商品ID,和商品ID对应的库存量,秒杀成功就将此商品库存量-1。现在假设有1000个线程来秒杀两件商品,500个线程秒杀第一个商品,500个线程秒杀第二个商品。我们来根据这个简单的业务场景来解释一下分布式锁。

通常具有秒杀场景的业务系统都比较复杂,承载的业务量非常巨大,并发量也很高。这样的系统往往采用分布式的架构来均衡负载。那么这1000个并发就会是从不同的地方过来,商品库存就是共享的资源,也是这1000个并发争抢的资源,这个时候我们需要将并发互斥管理起来。这就是分布式锁的应用。

1.实现分布式锁的几种方案

    1.Redis实现   (推荐)
    2.Zookeeper实现
    3.数据库实现

Redis实现分布式锁** 在集群等多服务器中经常使用到同步处理一下业务,这是普通的事务是满足不了业务需求,需要分布式锁** 分布式锁的常用3种实现:*        0.数据库乐观锁实现*        1.Redis实现  --- 使用redis的setnx()、get()、getset()方法,用于分布式锁,解决死锁问题*        2.Zookeeper实现*           参考:http://surlymo.iteye.com/blog/2082684*              ///article/103617.htm*              http://www.hollischuang.com/archives/1716?utm_source=tuicool&utm_medium=referral*          1、实现原理:基于zookeeper瞬时有序节点实现的分布式锁,其主要逻辑如下(该图来自于IBM网站)。大致思想即为:每个客户端对某个功能加锁时,在zookeeper上的与该功能对应的指定节点的目录下,生成一个唯一的瞬时有序节点。判断是否获取锁的方式很简单,只需要判断有序节点中序号最小的一个。当释放锁的时候,只需将这个瞬时节点删除即可。同时,其可以避免服务宕机导致的锁无法释放,而产生的死锁问题。2、优点锁安全性高,zk可持久化3、缺点性能开销比较高。因为其需要动态产生、销毁瞬时节点来实现锁功能。4、实现可以直接采用zookeeper第三方库curator即可方便地实现分布式锁** Redis实现分布式锁的原理:*  1.通过setnx(lock_timeout)实现,如果设置了锁返回1, 已经有值没有设置成功返回0*  2.死锁问题:通过实践来判断是否过期,如果已经过期,获取到过期时间get(lockKey),然后getset(lock_timeout)判断是否和get相同,*   相同则证明已经加锁成功,因为可能导致多线程同时执行getset(lock_timeout)方法,这可能导致多线程都只需getset后,对于判断加锁成功的线程,*   再加expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS)过期时间,防止多个线程同时叠加时间,导致锁时效时间翻倍*  3.针对集群服务器时间不一致问题,可以调用redis的time()获取当前时间

2.Redis分分布式锁的代码实现

  1.定义锁接口

package com.jay.service.redis;  /**  * Redis分布式锁接口  * Created by hetiewei on 2017/4/7.  */ public interface RedisDistributionLock {   /**    * 加锁成功,返回加锁时间    * @param lockKey    * @param threadName    * @return    */   public long lock(String lockKey, String threadName);    /**    * 解锁, 需要更新加锁时间,判断是否有权限    * @param lockKey    * @param lockValue    * @param threadName    */   public void unlock(String lockKey, long lockValue, String threadName);    /**    * 多服务器集群,使用下面的方法,代替System.currentTimeMillis(),获取redis时间,避免多服务的时间不一致问题!!!    * @return    */   public long currtTimeForRedis(); } 

   2.定义锁实现

package com.jay.service.redis.impl;  import com.jay.service.redis.RedisDistributionLock; 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.StringRedisTemplate; import org.springframework.data.redis.serializer.RedisSerializer;  import java.util.concurrent.TimeUnit;  /**  * Created by hetiewei on 2017/4/7.  */ public class RedisLockImpl implements RedisDistributionLock {    //加锁超时时间,单位毫秒, 即:加锁时间内执行完操作,如果未完成会有并发现象   private static final long LOCK_TIMEOUT = 5*1000;    private static final Logger LOG = LoggerFactory.getLogger(RedisLockImpl.class);    private StringRedisTemplate redisTemplate;    public RedisLockImpl(StringRedisTemplate redisTemplate) {     this.redisTemplate = redisTemplate;   }    /**    * 加锁    * 取到锁加锁,取不到锁一直等待知道获得锁    * @param lockKey    * @param threadName    * @return    */   @Override   public synchronized long lock(String lockKey, String threadName) {     LOG.info(threadName+"开始执行加锁");     while (true){ //循环获取锁       //锁时间       Long lock_timeout = currtTimeForRedis()+ LOCK_TIMEOUT +1;       if (redisTemplate.execute(new RedisCallback<Boolean>() {         @Override         public Boolean doInRedis(RedisConnection redisConnection) throws DataAccessException {           //定义序列化方式           RedisSerializer<String> serializer = redisTemplate.getStringSerializer();           byte[] value = serializer.serialize(lock_timeout.toString());           boolean flag = redisConnection.setNX(lockKey.getBytes(), value);           return flag;         }       })){         //如果加锁成功         LOG.info(threadName +"加锁成功 ++++ 111111");         //设置超时时间,释放内存         redisTemplate.expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS);         return lock_timeout;       }else {         //获取redis里面的时间         String result = redisTemplate.opsForValue().get(lockKey);         Long currt_lock_timeout_str = result==null?null:Long.parseLong(result);         //锁已经失效         if (currt_lock_timeout_str != null && currt_lock_timeout_str < System.currentTimeMillis()){           //判断是否为空,不为空时,说明已经失效,如果被其他线程设置了值,则第二个条件判断无法执行           //获取上一个锁到期时间,并设置现在的锁到期时间           Long old_lock_timeout_Str = Long.valueOf(redisTemplate.opsForValue().getAndSet(lockKey, lock_timeout.toString()));           if (old_lock_timeout_Str != null && old_lock_timeout_Str.equals(currt_lock_timeout_str)){             //多线程运行时,多个线程签好都到了这里,但只有一个线程的设置值和当前值相同,它才有权利获取锁             LOG.info(threadName + "加锁成功 ++++ 22222");             //设置超时间,释放内存             redisTemplate.expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS);              //返回加锁时间             return lock_timeout;           }         }       }        try {         LOG.info(threadName +"等待加锁, 睡眠100毫秒"); //        TimeUnit.MILLISECONDS.sleep(100);         TimeUnit.MILLISECONDS.sleep(200);       } catch (InterruptedException e) {         e.printStackTrace();       }     }   }    /**    * 解锁    * @param lockKey    * @param lockValue    * @param threadName    */   @Override   public synchronized void unlock(String lockKey, long lockValue, String threadName) {     LOG.info(threadName + "执行解锁==========");//正常直接删除 如果异常关闭判断加锁会判断过期时间     //获取redis中设置的时间     String result = redisTemplate.opsForValue().get(lockKey);     Long currt_lock_timeout_str = result ==null?null:Long.valueOf(result);      //如果是加锁者,则删除锁, 如果不是,则等待自动过期,重新竞争加锁     if (currt_lock_timeout_str !=null && currt_lock_timeout_str == lockValue){       redisTemplate.delete(lockKey);       LOG.info(threadName + "解锁成功------------------");     }   }    /**    * 多服务器集群,使用下面的方法,代替System.currentTimeMillis(),获取redis时间,避免多服务的时间不一致问题!!!    * @return    */   @Override   public long currtTimeForRedis(){     return redisTemplate.execute(new RedisCallback<Long>() {       @Override       public Long doInRedis(RedisConnection redisConnection) throws DataAccessException {         return redisConnection.time();       }     });   }  } 

  3.分布式锁验证     

@RestController @RequestMapping("/distribution/redis") public class RedisLockController {    private static final String LOCK_NO = "redis_distribution_lock_no_";    private static int i = 0;    private ExecutorService service;    @Autowired   private StringRedisTemplate redisTemplate;    /**    * 模拟1000个线程同时执行业务,修改资源    *    * 使用线程池定义了20个线程    *    */   @GetMapping("lock1")   public void testRedisDistributionLock1(){      service = Executors.newFixedThreadPool(20);      for (int i=0;i<1000;i++){       service.execute(new Runnable() {         @Override         public void run() {           task(Thread.currentThread().getName());         }       });     }    }    @GetMapping("/{key}")   public String getValue(@PathVariable("key") String key){     Serializable result = redisTemplate.opsForValue().get(key);     return result.toString();   }    private void task(String name) { //    System.out.println(name + "任务执行中"+(i++));      //创建一个redis分布式锁     RedisLockImpl redisLock = new RedisLockImpl(redisTemplate);     //加锁时间     Long lockTime;     if ((lockTime = redisLock.lock((LOCK_NO+1)+"", name))!=null){       //开始执行任务       System.out.println(name + "任务执行中"+(i++));       //任务执行完毕 关闭锁       redisLock.unlock((LOCK_NO+1)+"", lockTime, name);     }    } } 

4.结果验证:

      在Controller中模拟了1000个线程,通过线程池方式提交,每次20个线程抢占分布式锁,抢到分布式锁的执行代码,没抢到的等待

     结果如下:

2017-04-07 16:27:17.385 INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-4等待加锁, 睡眠100毫秒2017-04-07 16:27:17.385 INFO 8652 --- [pool-2-thread-7] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-7解锁成功------------------    2017-04-07 16:27:17.391 INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-5加锁成功 ++++ 111111pool-2-thread-5任务执行中9942017-04-07 16:27:17.391 INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-5执行解锁==========    2017-04-07 16:27:17.391 INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-1等待加锁, 睡眠100毫秒2017-04-07 16:27:17.391 INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-5解锁成功------------------    2017-04-07 16:27:17.397 INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-6加锁成功 ++++ 111111pool-2-thread-6任务执行中9952017-04-07 16:27:17.398 INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-6执行解锁==========    2017-04-07 16:27:17.398 INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-6解锁成功------------------    2017-04-07 16:27:17.400 INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-19加锁成功 ++++ 111111pool-2-thread-19任务执行中9962017-04-07 16:27:17.400 INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-19执行解锁==========    2017-04-07 16:27:17.400 INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-19解锁成功------------------    2017-04-07 16:27:17.571 INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-11加锁成功 ++++ 111111pool-2-thread-11任务执行中9972017-04-07 16:27:17.572 INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-11执行解锁==========    2017-04-07 16:27:17.572 INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-11解锁成功------------------    2017-04-07 16:27:17.585 INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-4加锁成功 ++++ 111111pool-2-thread-4任务执行中9982017-04-07 16:27:17.586 INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-4执行解锁==========    2017-04-07 16:27:17.586 INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-4解锁成功------------------    2017-04-07 16:27:17.591 INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-1加锁成功 ++++ 111111pool-2-thread-1任务执行中9992017-04-07 16:27:17.591 INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-1执行解锁==========    2017-04-07 16:27:17.591 INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-1解锁成功------------------

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


  • 上一条:
    Redis实现分布式锁和等待序列的方法示例
    下一条:
    Redis实现分布式锁的方法示例
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 在Redis中能实现的功能、常见应用介绍(0个评论)
    • 2024年Redis面试题之一(0个评论)
    • 在redis缓存常见出错及解决方案(0个评论)
    • 在redis中三种特殊数据类型:地理位置、基数(cardinality)估计、位图(Bitmap)使用场景介绍浅析(2个评论)
    • Redis 删除 key用 del 和 unlink 有啥区别?(1个评论)
    • 近期文章
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2017-12
    • 2020-03
    • 2020-05
    • 2021-04
    • 2022-03
    • 2022-05
    • 2022-08
    • 2023-02
    • 2023-04
    • 2023-07
    • 2024-01
    • 2024-02
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客