[toc]
今天我们来聊聊分布式锁。
使用场景
首先,我们看这样一个场景:客户下单的时候,我们调用库存中心进行减库存,那我们一般的操作都是
update store set num = $num where id=$id
AI 代码解读
这种通过设置库存的修改方式,我们知道在并发量高的时候会存在数据库的丢失更新,比如a,b当前两个事务,查询出来的库存都是5,a买了3个单子要把库存设置为2,而b买了1个单子要把库存设置为4,那这个时候就会出现a会覆盖b的更新,所以我们更多的都是会加个条件
update store set num = $num where id=$id and num=$query_num
AI 代码解读
即乐观锁的方式来处理,当然也可以通过版本号来处理乐观锁,都是一样的,但是这是更新一个表,如果我们牵扯到多个表呢,我们希望和这个单子关联的所有的表同一时间只能被一个线程来处理更新,多个线程按照不同的顺序去更新同一个单子关联的不同数据,出现死锁的概率比较大。对于非敏感的数据,我们也没有必要去都加乐观锁处理,我们的服务都是多机器部署的,要保证多进程多线程同时只能有一个进程的一个线程去处理,这个时候我们就需要用到分布式锁。
分布式锁的实现方式有很多,我们今天分别通过数据库,zk,redis以及tair的实现逻辑
数据库实现
加x锁
更新一个单子关联的所有的数据,先查询出这个单子,并加上排他锁,在进行一系列的更新操作
begin transaction;
select ...for update;
doSomething();
commit();
AI 代码解读
这种处理需要主要依靠排他锁来阻塞其他线程,不过这个需要注意几点:
- 查询的数据一定要在数据库里存在,如果不存在的话,数据库会加gap锁,而gap锁之间是兼容的,这种如果两个线程都加了gap锁,另一个再更新的话会出现死锁。不过一般能更新的数据都是存在的
- 后续的处理流程需要尽可能的时间短,即在更新的时候提前准备好数据,保证事务处理的时间足够的短,流程足够的短,因为开启事务是一直占着连接的,如果流程比较长会消耗过多的数据库连接的。
唯一键
通过在一张表里创建唯一键来获取锁,比如执行saveStore这个方法
insert table lock_store ('method_name') values($method_name)
AI 代码解读
其中method_name是个唯一键,通过这种方式也可以做到,解锁的时候直接删除改行记录就行。不过这种方式,锁就不会是阻塞式的,因为插入数据是立马可以得到返回结果的。
那针对以上数据库实现的两种分布式锁,存在什么样的优缺点呢
优点
- 简单,方便,快速实现
缺点
- 基于数据库,开销比较大,性能可能会存在影响
- 基于数据库的当前读来实现,数据库会在底层做优化,可能用到索引,可能不用到索引,这个依赖于查询计划的分析
zk的实现
使用zk来实现,代码网上比较多,我这里大致说下步骤,我们重点看redis的实现。
获取锁
- 先有一个锁跟节点,lockRootNode,这可以是一个永久的节点
- 客户端获取锁,先在lockRootNode下创建一个顺序的瞬时节点,保证客户端断开连接,节点也自动删除
- 调用lockRootNode父节点的getChildren()方法,获取所有的节点,并从小到大排序,如果创建的最小的节点是当前节点,则返回true,获取锁成功,否则,关注比自己序号小的节点的释放动作(exist watch),这样可以保证每一个客户端只需要关注一个节点,不需要关注所有的节点,避免羊群效应。
- 如果有节点释放操作,重复步骤3
释放锁
只需要删除步骤2中创建的节点即可
使用zk的分布式锁存在什么样的优缺点呢?
优点
- 客户端如果出现宕机故障的话,锁可以马上释放
- 可以实现阻塞式锁,通过watcher监听,实现起来也比较简单
- 集群模式,稳定性比较高
缺点
- 一旦网络有任何的抖动,zk就会认为客户端已经宕机,就会断掉连接,其他客户端就可以获取到锁。当然zk有重试机制,这个就比较依赖于其重试机制的策略了
- 自己没有做过测试,网上看到的说是性能上不如缓存
redis实现
分布式锁介绍这块,我们重点看下redis的分布式锁的实现。
我们先举个例子,比如现在我要更新产品的信息,产品的唯一键就是productId
简单实现1
public boolean lock(String key, V v, int expireTime){
int retry = 0;
//获取锁失败最多尝试10次
while (retry < failRetryTimes){
//获取锁
Boolean result = redis.setNx(key, v, expireTime);
if (result){
return true;
}
try {
//获取锁失败间隔一段时间重试
TimeUnit.MILLISECONDS.sleep(sleepInterval);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
return false;
}
}
return false;
}
public boolean unlock(String key){
return redis.delete(key);
}
public static void main(String[] args) {
Integer productId = 324324;
RedisLock<Integer> redisLock = new RedisLock<Integer>();
redisLock.lock(productId+"", productId, 1000);
}
AI 代码解读
这是一个简单的实现,存在的问题
- 可能会导致当前线程的锁误被其他线程释放,比如a线程获取到了锁正在执行,但是由于内部流程处理超时或者gc导致锁过期,这个时候b线程获取到了锁,a和b线程处理的是同一个productId,b还在处理的过程中,这个时候a处理完了,a去释放锁,可能就会导致a把b获取的锁释放了。
- 不能实现可重入
- 客户端如果第一次已经设置成功,但是由于超时返回失败,此后客户端尝试会一直失败
针对以上问题我们改进下
- v传requestId,然后我们在释放锁的时候判断一下,如果是当前requestId,那就可以释放,否则不允许释放
- 加入count的锁计数,在获取锁的时候查询一次,如果是当前线程已经持有的锁,那锁技术加1,直接返回true
简单实现2
private static volatile int count = 0;
public boolean lock(String key, V v, int expireTime){
int retry = 0;
//获取锁失败最多尝试10次
while (retry < failRetryTimes){
//1.先获取锁,如果是当前线程已经持有,则直接返回
//2.防止后面设置锁超时,其实是设置成功,而网络超时导致客户端返回失败,所以获取锁之前需要查询一下
V value = redis.get(key);
//如果当前锁存在,并且属于当前线程持有,则锁计数+1,直接返回
if (null != value && value.equals(v)){
count ++;
return true;
}
//如果锁已经被持有了,那需要等待锁的释放
if (value == null || count <= 0){
//获取锁
Boolean result = redis.setNx(key, v, expireTime);
if (result){
count = 1;
return true;
}
}
try {
//获取锁失败间隔一段时间重试
TimeUnit.MILLISECONDS.sleep(sleepInterval);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
return false;
}
}
return false;
}
public boolean unlock(String key, String requestId){
String value = redis.get(key);
if (Strings.isNullOrEmpty(value)){
count = 0;
return true;
}
//判断当前锁的持有者是否是当前线程,如果是的话释放锁,不是的话返回false
if (value.equals(requestId)){
if (count > 1){
count -- ;
return true;
}
boolean delete = redis.delete(key);
if (delete){
count = 0;
}
return delete;
}
return false;
}
public static void main(String[] args) {
Integer productId = 324324;
RedisLock<String> redisLock = new RedisLock<String>();
String requestId = UUID.randomUUID().toString();
redisLock.lock(productId+"", requestId, 1000);
}
AI 代码解读
这种实现基本解决了误释放和可重入的问题。
这里说明几点:
- 引入count实现重入的话,看业务需要,并且在释放锁的时候,其实也可以直接就把锁删除了,一次释放搞定,不需要在通过count数量释放多次,看业务需要吧
- 关于要考虑设置锁超时,所以需要在设置锁的时候查询一次,可能会有性能的考量,看具体业务吧
- 目前获取锁失败的等待时间是在代码里面设置的,可以提出来,修改下等待的逻辑即可
错误实现
之前在网上还看到有这种实现方式,就是获取到锁之后要检查下锁的过期时间,如果锁过期了要重新设置下时间,大致代码如下
public boolean tryLock2(String key, int expireTime){
long expires = System.currentTimeMillis() + expireTime;
//获取锁
Boolean result = redis.setNx(key, expires, expireTime);
if (result){
return true;
}
V value = redis.get(key);
if (value != null && (Long)value < System.currentTimeMillis()){
//锁已经过期
String oldValue = redis.getSet(key, expireTime);
if (oldValue != null && oldValue.equals(value)){
return true;
}
}
return false;
}
AI 代码解读
这种实现存在的问题,过度依赖当前服务器的时间了,如果在大量的并发请求下,都判断出了锁过期,而这个时候再去设置锁的时候,最终是会只有一个线程,但是可能会导致不同服务器根据自身不同的时间覆盖掉最终获取锁的那个线程设置的时间。
tair的实现
通过tair来实现分布式锁和redis的实现核心差不多,不过tair有个很方便的api,感觉是实现分布式锁的最佳配置,就是Put api调用的时候需要传入一个version,就和数据库的乐观锁一样,修改数据之后,版本会自动累加,如果传入的版本和当前数据版本不一致,就不允许修改,具体可以看下这篇文章的实现:Tair分布式锁这里就不再多说了
参考
http://www.cnblogs.com/luxiaoxun/p/4889764.html
http://blog.csdn.net/abccheng/article/details/72420996