[Golang 修仙之路] 分布式专题:分布式锁

分析

锁的基本要求(无论是分布式锁还是单机锁):

  1. 要有锁的存储空间。
  2. 锁要被唯一标识。
  3. 锁要有2种状态。

但是这不够,还需要考虑3个问题:

  1. 判断锁状态 和 上锁 2步操作的原子性。
  2. 锁需要被及时释放。
    • (主机A获取到锁,A宕机,一直不释放锁,资源被占用)
  3. 释放锁时,需要确认这锁是自己持有的。
    • (A获取锁,A挂了,挂之前及时释放锁,B一看没锁,B获取锁成功,A重启,释放锁,B一看md自己的锁被别人释放了)

进阶:

  1. 惊群效应(Herd Effect):在分布式锁中,惊群效应指的是,在有多个请求等待获取锁的时候,一旦占有锁的线程释放之后,如果所有等待的方都同时被唤醒,尝试抢占锁。但是这样的情况会造成比较大的开销,那么在实现分布式锁的时候,应该尽量避免惊群效应的产生。

Redis 实现

过程

获取锁:

arduino 复制代码
set lock_key <setter_id> nx ex 10
  • nx 保证了获取锁和加锁2步的原子性。
  • ex 保证了锁不会被一直持有。
  • setter_id 保证了锁被释放时可以检查是不是自己的。

释放锁:

  1. 判断锁的value是不是自己的setter_id,是才释放锁。
  2. del lock_key

这两步操作的原子性通过lua脚本保证。

lua 复制代码
if (redis.call('GET', KEY[1]) == ARGV[1]) then
    return redis.call('DEL', KEY[1])
else 
    return 0

问题

Redis 主从同步是异步的,主节点set nx成功,从节点还没同步,此时其他进程想获取锁,走从节点读,仍然能获取到。

zookeeper实现

过程

ZooKeeper(以下简称"ZK")中有一种节点叫做顺序节点,假如我们在/lock/目录下创建3个节点,ZK集群会按照发起创建的顺序来创建节点,节点分别为/lock/0000000001、/lock/0000000002、/lock/0000000003。

ZK中还有一种名为临时节点的节点,临时节点由某个客户端创建,当客户端与ZK集群断开连接,则该节点自动被删除。EPHEMERAL_SEQUENTIAL为临时顺序节点。

根据ZK中节点是否存在,可以作为分布式锁的锁状态,以此来实现一个分布式锁,下面是分布式锁的基本逻辑:

  1. 客户端调用create()方法创建名为"/dlm-locks/lockname/lock-"的临时顺序节点。
  2. 客户端调用getChildren("lockname")方法来获取所有已经创建的子节点。
  3. 客户端获取到所有子节点path之后,如果发现自己在步骤1中创建的节点是所有节点中序号最小的,那么就认为这个客户端获得了锁。
  4. 如果创建的节点不是所有节点中需要最小的,那么则监视比自己创建节点的序列号小的最大的节点,进入等待。直到下次监视的子节点变更的时候,再进行子节点的获取,判断是否获取锁。

释放锁的过程相对比较简单,就是删除自己创建的那个子节点即可,不过也仍需要考虑删除节点失败等异常情况。

优势

  • zk的临时节点可以直接避免网络断开或主机宕机,锁状态无法清除的问题。
  • 顺序节点可以避免惊群效应。

这些特性都使得利用ZK实现分布式锁成为了最普遍的方案之一。

相关推荐
非凡ghost2 分钟前
PixPin截图工具(支持截长图截动图) 中文绿色版
前端·javascript·后端
武子康3 分钟前
大数据-133 ClickHouse 概念与基础|为什么快?列式 + 向量化 + MergeTree 对比
大数据·后端·nosql
脚踏实地的大梦想家18 分钟前
【Go】P11 掌握 Go 语言函数(二):进阶玩转高阶函数、闭包与 Defer/Panic/Recover
开发语言·后端·golang
在未来等你36 分钟前
Kafka面试精讲 Day 29:版本升级与平滑迁移
大数据·分布式·面试·kafka·消息队列
用户68545375977691 小时前
🔥 服务熔断降级:微服务的"保险丝"大作战!
后端
Tech有道1 小时前
拼多多「面试官问我:LRU 和 LFU 你选谁?」我:看场景啊哥!😂
后端
用户68545375977691 小时前
🎬 开场:RPC框架的前世今生
后端
在未来等你1 小时前
Kafka面试精讲 Day 30:Kafka面试真题解析与答题技巧
大数据·分布式·面试·kafka·消息队列
王中阳Go背后的男人1 小时前
Docker磁盘满了?这样清理高效又安全
后端·docker
用户68545375977691 小时前
🎛️ 分布式配置中心:让配置管理不再是噩梦!
后端