redis实现分布式锁需要考虑的因素以及可重入锁实现
死锁
错误例子
解决方式
锁超时
错误例子
String lockKey="stock:product:1";
boolean isGetLock=false;
try{
//假设是原子性的 获取锁并设置锁10秒
isGetLock==setnx(lockKey,10);
if(!isGetLock){
throw new Exception("系统繁忙!请稍后再试");
}
//模拟需要执行12秒
Thread.sleep(12);
}finally {
if(isGetLock){
del(lockKey);
}
}
假设有线程A线程B 2个线程
线程A率先拿到锁因为我们设置的锁10秒自动释放(redis过期时间10秒) 而我们程序需要执行10秒以上
10.1ms秒的时候线程B进来 因为redis锁key已经过期成功拿到锁 并阻塞在12秒处
12秒后线程A 执行完 执行del操作 导致释放了线程B的锁
解决方式1
String lockKey="stock:product:1";
boolean isGetLock=false;
//用来标识当前身份
String currentIndex=UUID.randomUUID().toString();
try{
//假设是原子性的 获取锁并设置锁10秒 同时设置一个值为currentIndex
isGetLock==setnx(lockKey,currentIndex,10);
if(!isGetLock){
throw new Exception("系统繁忙!请稍后再试");
}
//模拟需要执行12秒
Thread.sleep(12);
}finally {
if(isGetLock){
String lockValue=get(lockKey);
//表示是当前线程的锁 释放
if(lockValue!=null&&lockValue.equals(currentIndex)) {
del(lockKey);
}
}
}
方式1优化方案
简单一看 好像并没有什么问题 但是需要注意 get 比较 和del并不是原子性的
比如 线程A get完之后 lockkey因为超时释放 线程B 成功获得锁 线程A再执行if判断 会删除调线程B的锁
改为lua脚本
if redis.call("get",KEYS[]==ARGV[]) then
return redis.call("del","KEYS1")
else
return
end
主从切换
可重入锁实现
/**
* @Auther: liqiang
* @Date: 2019/7/14 14:59
* @Description:
*/
public class RedisWithReentrantLock {
private ThreadLocal<Map<String,Integer>> lockers=new ThreadLocal<>();
private Jedis jedis;
public RedisWithReentrantLock(Jedis jedis){
this.jedis=jedis;
}
/**
* 加锁
*/
private boolean _lock(String key){
String value=String.valueOf(System.currentTimeMillis());;
return jedis.set(key,value,"nx","ex",5L)!=null;
}
/**
* 释放锁
* @param key
*/
private void _unlock(String key){
jedis.del(key);
} /**
* 从线程缓存获取map 没有就初始化一个
* @return
*/
private Map<String,Integer> currentLockers(){
Map<String,Integer> refs=lockers.get();
if(refs==null){
refs=new HashMap<String,Integer>();
lockers.set(refs);
}
return lockers.get();
} /**
* 可重入锁
* @param key
* @return
*/
public boolean lock(String key){
/**
* 选择map的原因是 一个线程里面可能有很多加锁的地方
*/
Map<String,Integer> lockers=currentLockers();
/**
*如果存在 表示是重入加锁
*/
if(lockers.containsKey(key)){
lockers.put(key,lockers.get(key)+1);
//延长过期时间
jedis.expire(key,5000);
return true;
}
//走到这里表示是头部第一次加锁 加锁并对应map数量+1
boolean isGetLock=_lock(key);
lockers.put(key,1);
return isGetLock;
} /**
* 释放锁
* @param key
* @return
*/
public boolean unLock(String key){
/**
* 获得map
*/
Map<String,Integer> lockers=currentLockers();
/**
* 表示key未加过锁 或者释放了
*/
Integer refCnt=lockers.get(key);
if(refCnt==null){
return false;
}
//-1
refCnt-=1;
//大于0表示不是头部锁释放
if(refCnt>0){
lockers.put(key,refCnt);
}else{
//小于等于0 表示是头部锁释放 删除mapkey
lockers.remove(key);
/**
* 释放锁
*/
_unlock(key);
}
return true;
}
public static void main(String[] args) {
Jedis conn = new Jedis("127.0.0.1",6379);
conn.select(1);
RedisWithReentrantLock redisWithReentrantLock=new RedisWithReentrantLock(conn);
String lockKey="lock:key3";
redisWithReentrantLock.lock(lockKey);
redisWithReentrantLock.lock(lockKey); redisWithReentrantLock.unLock(lockKey);
redisWithReentrantLock.unLock(lockKey);
}
}
一些建议
建议涉及并发的地方能用原子性操作就用原子性
例子一
tock stock=stockDao.get(id);
if(stock.getNumber()-10<0){
throw new Exception("库存不足");
}
stock.setNumber(stock.getNumber-10);
stockDao.update(stock);
这种情况就算加锁的情况 如果出现上面说的几种极端情况 或者锁失效了 会导致超卖以及库存异常问题
优化方案
Stock stock=stockDao.get(id);
/**
* 这里可能会疑惑 下面有原子性的update加 where校验超卖 这一步是否不需要了
* 个人理解 程序进行校验 总比全部堆到数据库校验好的多
* 比如库存卖完了 还持续有并发请求 在这里就可以全部挡在外面
*/
if(stock.getNumber()-10<0){
throw new Exception("库存不足");
}
stock.setNumber(stock.getNumber-10);
//原子性的update
Integer updateNumber=stockDao.excuteSql("update stock set number-=10 where id=:id and number>=0",id);
//表示未能成功修改
if(updateNumber<=0){
throw new Exception("库存不足");
}
redis则使用对应redis递增递减
对于提供给管理员的库存盘点 也是使用原子性递增递减
盘增
比如当前库存是10 管理员调整20 则是+10 而不要直接set 20 不然并发时 10 卖了5 这个时候20才提交 则变成了20 如果+10 则变成15
盘减
比如当前库存是10 管理员 需要调整为5 并发时减成了0 执行update stock set number-=5 where id=:id and number>=0 number>=0并不成立所以修改失败
高并发时建议(比如秒杀场景)
将库存全量到redis 通过Incrby 命令实现原子性递增递减 如果消息发送失败需要进行补偿
update stock set number-=10 where id=:id and number>=0 通过mq 队列异步执行 否则会出现同一个库存并发改 部分是失败数据库抛出waitLock tps就上不去 还会有大量请求到数据库 可能把redis
弄挂
redis实现分布式锁需要考虑的因素以及可重入锁实现的更多相关文章
- Redisson 分布式锁源码 01:可重入锁加锁
前言 相信小伙伴都是使用分布式服务,那一定绕不开分布式服务中数据并发更新问题! 单系统很容易想到 Java 的各种锁,像 synchronize.ReentrantLock 等等等,那分布式系统如何处 ...
- redis分布式锁-可重入锁
redis分布式锁-可重入锁 上篇redis实现的分布式锁,有一个问题,它不可重入. 所谓不可重入锁,即若当前线程执行某个方法已经获取了该锁,那么在方法中尝试再次获取锁时,就会获取不到被阻塞. 同一个 ...
- Java并发编程:自己动手写一把可重入锁
关于线程安全的例子,我前面的文章Java并发编程:线程安全和ThreadLocal里面提到了,简而言之就是多个线程在同时访问或修改公共资源的时候,由于不同线程抢占公共资源而导致的结果不确定性,就是在并 ...
- java高并发系列 - 第12天JUC:ReentrantLock重入锁
java高并发系列 - 第12天JUC:ReentrantLock重入锁 本篇文章开始将juc中常用的一些类,估计会有十来篇. synchronized的局限性 synchronized是java内置 ...
- 浅谈Java中的锁:Synchronized、重入锁、读写锁
Java开发必须要掌握的知识点就包括如何使用锁在多线程的环境下控制对资源的访问限制 ◆ Synchronized ◆ 首先我们来看一段简单的代码: 12345678910111213141516171 ...
- synchronized 是可重入锁吗?为什么?
什么是可重入锁? 关于什么是可重入锁,我们先来看一段维基百科的定义. 若一个程序或子程序可以“在任意时刻被中断然后操作系统调度执行另外一段代码,这段代码又调用了该子程序不会出错”,则称其为可重入(re ...
- JAVA锁机制-可重入锁,可中断锁,公平锁,读写锁,自旋锁,
如果需要查看具体的synchronized和lock的实现原理,请参考:解决多线程安全问题-无非两个方法synchronized和lock 具体原理(百度) 在并发编程中,经常遇到多个线程访问同一个 ...
- Java 多线程 重入锁
作为关键字synchronized的替代品(或者说是增强版),重入锁是synchronized的功能扩展.在JDK 1.5的早期版本中,重入锁的性能远远好于synchronized,但从JDK 1.6 ...
- synchronized的功能拓展:重入锁(读书笔记)
重入锁可以完全代替synchronized关键字.在JDK5.0的早期版本中,重入锁的性能远远好于synchronized,但是从JDK6.0开始.JDK在synchronized上做了大量的优化. ...
随机推荐
- 2018-2-13-win10-uwp-绑定密码
title author date CreateTime categories win10 uwp 绑定密码 lindexi 2018-2-13 17:23:3 +0800 2018-2-13 17: ...
- linux 系统磁盘管理体系
目录 linux 系统磁盘管理体系 一.磁盘的基本概念 二.磁盘的内部结构 三.磁盘的外部结构 四.磁盘的接口及类型 五.fdisk磁盘分区实践 六.gdisk 分区 七.parted 高级分区工具. ...
- tomcat manager详解
Tomcat Manager是Tomcat自带的.用于对Tomcat自身以及部署在Tomcat上的应用进行管理的web应用.Tomcat是Java领域使用最广泛的服务器之一,因此Tomcat Mana ...
- php递归无限分类、根据子类获取所有顶类
//递归无限分类树 public static function diGui($data, $pid) { $arr = collect([]); if (empty($data)) { return ...
- Devops、CI\CD、Jenkins
Devops DevOps对应用程序发布的影响 在很多企业中,应用程序发布是一项涉及多个团队.压力很大.风险很高的活动.然而在具备DevOps能力的组织中,应用程序发布的风险很低,原因如下 [2] : ...
- vue.js 分页
<template> <div class="index"> <el-pagination background :hide-on-single-pa ...
- 【leetcode】947. Most Stones Removed with Same Row or Column
题目如下: On a 2D plane, we place stones at some integer coordinate points. Each coordinate point may h ...
- <自动化测试>之<使用unittest Python测试框架进行参数化测试>
最近在看视频时,虫师简单提到了简化自动化测试脚本用例中的代码量,而python中本身的参数化方法用来测试很糟糕,他在实际操作中使用了parameterized参数化... 有兴趣就查了下使用的方法,来 ...
- BZOJ 3531: [Sdoi2014]旅行(树链剖分+线段树)
传送门 解题思路 以每个颜色为根开一棵权值线段树,下标就是\(dfs\)序,其余都是基本操作,要动态开点. 代码 #include<iostream> #include<cstdio ...
- MySQL 下载,安装,配置windows 服务
本次使用的是压缩包的方式是可以纯手动自己折腾各种配置... ok,闲话少叙,我们准备发车... 一.先要去mysql官网去下载压缩包咯 ①下载地址:https://dev.mysql.com/down ...