为什么要用分布式锁?

先上一张截图,这是在浏览别人的博客时看到的.

在了解为什么要用分布式锁之前,我们应该知道到底什么是分布式锁.

锁按照不同的维度,有多种分类.比如

1.悲观锁,乐观锁;

2.公平锁,非公平锁;

3.独享锁,共享锁;

4.线程锁,进程锁;

等等.

我们平时用的锁,比如 lock,它是线程锁,主要用来给方法,代码块加锁.由于进程的内存单元是被其所有线程共享的,所以线程锁控制的实际是多个线程对同一块内存区域的访问.

有线程锁,就必然有进程锁.顾名思义,进程锁的目的是控制多个进程对共享资源的访问.因为进程之间彼此独立,各个进程是无法控制其他进程对资源的访问,所以只能通过操作系统来控制.比如 Mutex.

但是进程锁有一个前提,那就是需要多个进程在同一个系统中,如果多个进程不在同一个系统,那就只能使用分布式锁来控制了.

分布式锁是控制分布式系统中不同系统之间访问共享资源的一种锁实现.它和线程锁,进程锁的作用都是一样,只是范围不一样.

所以要实现分布式锁,就必须依靠第三方存储介质来存储锁的信息.因为各个进程之间彼此谁都不服谁,只能找一个带头大哥咯;

以下示例需引用NUGET: CSRedisCore

示例一

CSRedisClient redisClient = new CSRedis.CSRedisClient("127.0.0.1:6379,defaultDatabase=0");var lockKey = "lockKey";var stock = 5;//商品库存var taskCount = 10;//线程数量redisClient.Del(lockKey);//测试前,先把锁删了.for (int i = 0; i < taskCount; i++){    Task.Run(() =>    {        //获取锁        do        {            //setnx : key不存在才会成功,存在则失败.            var success = redisClient.SetNx(lockKey, 1);            if (success == true)            {                break;            }            Thread.Sleep(TimeSpan.FromSeconds(1));//休息1秒再尝试获取锁        } while (true);        Console.WriteLine($"线程:{Task.CurrentId} 拿到了锁,开始消费");        if (stock <= 0)        {            Console.WriteLine($"库存不足,线程:{Task.CurrentId} 抢购失败!");            redisClient.Del(lockKey);            return;        }        stock--;        //模拟处理业务        Thread.Sleep(TimeSpan.FromSeconds(new Random().Next(1, 3)));        Console.WriteLine($"线程:{Task.CurrentId} 消费完毕!剩余 {stock} 个");        //业务处理完后,释放锁.        redisClient.Del(lockKey);    });}

运行结果:

看起来貌似没毛病,实际上上述代码有个致命的问题:

当某个线程拿到锁之后,如果系统崩溃了,那么锁永远都不会被释放.因此,我们应该给锁加一个过期时间,当时间到了,还没有被主动释放,我们就让redis释放掉它,以保证其他消费者可以拿到锁,进行消费.

这里给锁加过期时间也有讲究,不能拿到锁后再加,比如:

//setnx : key不存在才会成功,存在则失败.var success = redisClient.SetNx(lockKey, 1);if (success == true){    redisClient.Set(lockKey, 1, expireSeconds: 5);    break;}

这样操作的话,获取锁和设置锁的过期时间就不是原子操作,同样会出现上面提到的问题.Redis 提供了一个合而为一的操作可以解决这个问题.

//set : key存在则失败,不存在才会成功,并且过期时间5秒var success = redisClient.Set(lockKey, 1, expireSeconds: 5, exists: RedisExistence.Nx);

这个问题虽然解决了,但随之产生了一个新的问题:

假设有3个线程A,B,C

当线程A拿到锁后执行业务的时候超时了,超过了锁的过期时间还没执行完,这时候锁被Redis释放了,

于是线程B拿到了锁并开始执行业务逻辑.

当线程B的业务逻辑还没执行完的时候,线程A的业务逻辑执行完了,于是乎就跑去释放掉了锁.

这时候线程C就可以拿到锁开始执行它的业务逻辑.

这不就乱套了么...

因此,线程在释放锁的时候应该判断这个锁还属不属于自己.

所以,在设置锁的时候,redis的value值不能像上面代码那样,随便给个1,而应该给一个随机值,代表当前线程.

var id = Guid.NewGuid().ToString("N");//获取锁do{    //set : key存在则失败,不存在才会成功,并且过期时间5秒    var success = redisClient.Set(lockKey, id, expireSeconds: 5, exists: RedisExistence.Nx);    if (success == true)    {        break;    }    Thread.Sleep(TimeSpan.FromSeconds(1));//休息1秒再尝试获取锁} while (true);Console.WriteLine($"线程:{Task.CurrentId} 拿到了锁,开始消费");  .........//业务处理完后,释放锁.var value = redisClient.Get<string>(lockKey);if (value == id){    redisClient.Del(lockKey);}

完美了吗?

不完美.还是老生常谈的问题,取value和删除key 分了两步走,不是原子操作.

并且,这里还不能用pipe,因为需要根据取到的value来决定下一个操作.上面设置过期时间倒是可以用pipe.

所以,这里只能用lua.

2020.10.09 补:将库存放到redis

完整的代码如下:

CSRedisClient redisClient = new CSRedis.CSRedisClient("127.0.0.1:6379,defaultDatabase=0");var lockKey = "lockKey";var stockKey = "stock";redisClient.Set(stockKey, 5);//商品库存var releaseLockScript = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";//释放锁的redis脚本redisClient.Del(lockKey);//测试前,先把锁删了.Parallel.For(0, 10, i =>{    var id = Guid.NewGuid().ToString("N");    //获取锁    do    {        //set : key存在则失败,不存在才会成功,并且过期时间5秒        var success = redisClient.Set(lockKey, id, expireSeconds: 5, exists: RedisExistence.Nx);        if (success == true)        {            break;        }        Thread.Sleep(TimeSpan.FromSeconds(1));//休息1秒再尝试获取锁    } while (true);    Console.WriteLine($"线程:{Task.CurrentId} 拿到了锁,开始消费");    //扣减库存    var currentStock = redisClient.IncrBy(stockKey, -1);    if (currentStock < 0)    {        Console.WriteLine($"库存不足,线程:{Task.CurrentId} 抢购失败!");        redisClient.Eval(releaseLockScript, lockKey, id);        return;    }    //模拟处理业务,这里不考虑失败的情况    Thread.Sleep(TimeSpan.FromSeconds(new Random().Next(1, 3)));    Console.WriteLine($"线程:{Task.CurrentId} 消费完毕!剩余 {currentStock} 个");    //业务处理完后,释放锁.    redisClient.Eval(releaseLockScript, lockKey, id);});

这篇文章只介绍了单节点Redis的分布式锁,因为单节点,所以不是高可用.

多节点Redis则需要用官方介绍的RedLock,这玩意有点绕,我需要捋一捋.

c# 操作redisclient 设置过期时间_C# Redis分布式锁 单节点相关推荐

  1. c# 操作redisclient 设置过期时间_C# Redis分布式锁单节点

    (给DotNet加星标,提升.Net技能) 转自:热敷哥cnblogs.com/refuge/p/13774008.html 为什么要用分布式锁? 先上一张截图,这是在浏览别人的博客时看到的. 在了解 ...

  2. redis文档翻译_key设置过期时间

    Available since 1.0.0.    使用開始版本号1.01 Time complexity: O(1)  时间复杂度O(1) 出处:http://blog.csdn.net/colum ...

  3. Redis 分布式锁没这么简单,网上大多数都有 bug

    Redis 分布式锁这个话题似乎烂大街了,不管你是面试还是工作,随处可见,「码哥」为啥还写? 因为看过很多文章没有将分布式锁的各种问题讲明白,所以准备写一篇,也当做自己的学习总结. 在进入正文之前,我 ...

  4. 第六章 商品详情进阶 + redis分布式锁 + redis问题解决 + redisson + 布隆过滤器

    一.商品详情页面优化 1.1 思路 虽然咱们实现了页面需要的功能,但是考虑到该页面是被用户高频访问的,所以性能需要优化. 一般一个系统最大的性能瓶颈,就是数据库的io操作.从数据库入手也是调优性价比最 ...

  5. 聊聊redis分布式锁的8大坑

    在分布式系统中,由于redis分布式锁相对于更简单和高效,成为了分布式锁的首先,被我们用到了很多实际业务场景当中. 但不是说用了redis分布式锁,就可以高枕无忧了,如果没有用好或者用对,也会引来一些 ...

  6. 卧槽,redis分布式锁如果用不好,坑真多

    在分布式系统中,由于redis分布式锁相对于更简单和高效,成为了分布式锁的首先,被我们用到了很多实际业务场景当中. 但不是说用了redis分布式锁,就可以高枕无忧了,如果没有用好或者用对,也会引来一些 ...

  7. Redis分布式锁的实现以及原理

    1 前言 在程序中,我们想要保证一个变量的可见性及原子性,我们可以用volatile(对任意单个volatile变量的读/写具有原子性,但类似于volatile++这种复合操作不具有原子性).sync ...

  8. Java三种方式实现redis分布式锁

    一.引入原因 在分布式服务中,常常有如定时任务.库存更新这样的场景. 在定时任务中,如果不使用quartz这样的分布式定时工具,只是简单的使用定时器来进行定时任务,在服务分布式部署中,就有可能存在定时 ...

  9. Redis 分布式锁的正确实现原理演化历程与 Redisson 实战总结

    Redis 分布式锁使用 SET 指令就可以实现了么?在分布式领域 CAP 理论一直存在. 分布式锁的门道可没那么简单,我们在网上看到的分布式锁方案可能是有问题的. 一步步带你深入分布式锁是如何一步步 ...

最新文章

  1. C++ 实验 5.12
  2. build_transformer_model如果不返回keras的bert模型返回的是什么?
  3. golang中的strings.Split
  4. Unity3d(U3D) Windows/Android/IOS 播放rtmp/rtsp方案
  5. hibernate报错 net.sf.json.util.CycleDetectionStrategy$StrictionStrategyRepeatedReferenceAsObject
  6. 边工作边刷题:70天一遍leetcode: day 98
  7. 高空真人特技表演的这些冷知识,你都知道吗?
  8. 【转】解决XMLHTTP获取网页中文乱码问题
  9. r语言 array c函数,R语言 数组
  10. 吴恩达神经网络和深度学习-学习笔记-11-Momentum梯度下降法
  11. android 设置类PreferenceActivity
  12. 全面的软件测试-软件测试图解
  13. 教你安装ps,pr,ae,ai等Adobe软件,办公必备
  14. win10邮箱怎么设置qq邮箱服务器地址,老鸟给你说win10自带邮件怎么添加qq邮箱的解决方式...
  15. 武汉大学计算机学院2015级,武汉大学研究生课程-数据挖掘-2015级研究生试题.doc...
  16. HTML简单动画制作
  17. ale插件 vim_vim 撸码必备插件之 autoformat 与 ale[视频]
  18. 2.跟我走吧,现在就出发
  19. 阿里千万级实时监控系统技术揭秘TSDB时序业务场景
  20. WinVista发布前最大敌人是Win95??!!

热门文章

  1. HDU2571 命运【动态规划DP】
  2. HDU2098 分拆素数和【筛选法】
  3. 从 dig(nslookup) bind —— windows 下的域名解析服务器信息的查看
  4. 方阵的迹(trace)及其微分(导数)
  5. 推理集 —— 举一反三
  6. 商场内自动扶梯的研究
  7. java list api_Java核心API -- 6(Collection集合List、Set、ArrayList、HashSet)
  8. 计算机信息技术基础知识教案,计算机的基础知识
  9. Adapter中notify(),notifyAll(),notifyDataSetChanged(),notifyDataSetInvalidaded()方法的区别
  10. python3哪个版本稳定-Python 3.9 发布稳定版本,八大特性学起来!