Redis 实现同步锁
1、技术方案
1.1、redis的基本命令
1)SETNX命令(SET if Not eXists)
语法:SETNX key value
功能:当且仅当 key 不存在,将 key 的值设为 value ,并返回1;若给定的 key 已经存在,则 SETNX 不做任何动作,并返回0。
2)expire命令
语法:expire KEY seconds
功能:设置key的过期时间。如果key已过期,将会被自动删除。
3)DEL命令
语法:DEL key [KEY …]
功能:删除给定的一个或多个 key ,不存在的 key 会被忽略。
1.2、实现同步锁原理
1)加锁:“锁”就是一个存储在redis里的key-value对,key是把一组投资操作用字符串来形成唯一标识,value其实并不重要,因为只要这个唯一的key-value存在,就表示这个操作已经上锁。
2)解锁:既然key-value对存在就表示上锁,那么释放锁就自然是在redis里删除key-value对。
3)阻塞、非阻塞:阻塞式的实现,若线程发现已经上锁,会在特定时间内轮询锁。非阻塞式的实现,若发现线程已经上锁,则直接返回。
4)处理异常情况:假设当投资操作调用其他平台接口出现等待时,自然没有释放锁,这种情况下加入锁超时机制,用redis的expire命令为key设置超时时长,过了超时时间redis就会将这个key自动删除,即强制释放锁
(此步骤需在JAVA内部设置同样的超时机制,内部超时时长应小于或等于redis超时时长)。
1.3、处理流程图
2、代码实现
2.1、同步锁工具类
1 package com.mic.synchrolock.util;2 3 import java.util.ArrayList;4 import java.util.List;5 import java.util.UUID;6 7 import javax.annotation.PostConstruct;8 import javax.annotation.PreDestroy;9 10 import org.apache.commons.logging.Log;11 import org.apache.commons.logging.LogFactory;12 13 import org.springframework.beans.factory.annotation.Autowired;14 15 import com.mic.constants.Constants;16 import com.mic.constants.InvestType;17 18 /**19 * 分布式同步锁工具类20 * @author Administrator21 *22 */23 public class SynchrolockUtil { 24 25 private final Log logger = LogFactory.getLog(getClass());26 27 @Autowired28 private RedisClientTemplate redisClientTemplate;29 30 public final String RETRYTYPE_WAIT = "1"; //加锁方法当对象已加锁时,设置为等待并轮询31 public final String RETRYTYPE_NOWAIT = "0"; //加锁方法当对象已加锁时,设置为直接返回32 33 private String requestTimeOutName = ""; //投资同步锁请求超时时间34 private String retryIntervalName = ""; //投资同步锁轮询间隔35 private String keyTimeoutName = ""; //缓存中key的失效时间36 private String investProductSn = ""; //产品Sn37 private String uuid; //对象唯一标识38 39 private Long startTime = System.currentTimeMillis(); //首次调用时间40 public Long getStartTime() {41 return startTime;42 }43 44 List<String> keyList = new ArrayList<String>(); //缓存key的保存集合45 public List<String> getKeyList() {46 return keyList;47 }48 public void setKeyList(List<String> keyList) {49 this.keyList = keyList;50 }51 52 @PostConstruct53 public void init() {54 uuid = UUID.randomUUID().toString();55 }56 57 @PreDestroy58 public void destroy() {59 this.unlock();60 }61 62 63 /**64 * 根据传入key值,判断缓存中是否存在该key65 * 存在-已上锁:判断retryType,轮询超时,或直接返回,返回ture66 * 不存在-未上锁:将该放入缓存,返回false67 * @param key68 * @param retryType 当遇到上锁情况时 1:轮询;0:直接返回69 * @return70 */71 public boolean islocked(String key,String retryType){72 boolean flag = true;73 logger.info("====投资同步锁设置轮询间隔、请求超时时长、缓存key失效时长====");74 //投资同步锁轮询间隔 毫秒75 Long retryInterval = Long.parseLong(Constants.getProperty(retryIntervalName));76 //投资同步锁请求超时时间 毫秒77 Long requestTimeOut = Long.parseLong(Constants.getProperty(requestTimeOutName));78 //缓存中key的失效时间 秒79 Integer keyTimeout = Integer.parseInt(Constants.getProperty(keyTimeoutName));80 81 //调用缓存获取当前产品锁82 logger.info("====当前产品key为:"+key+"====");83 if(isLockedInRedis(key,keyTimeout)){84 if("1".equals(retryType)){85 //采用轮询方式等待86 while (true) {87 logger.info("====产品已被占用,开始轮询====");88 try {89 Thread.sleep(retryInterval);90 } catch (InterruptedException e) {91 logger.error("线程睡眠异常:"+e.getMessage(), e);92 return flag;93 }94 logger.info("====判断请求是否超时====");95 Long currentTime = System.currentTimeMillis(); //当前调用时间96 long Interval = currentTime - startTime;97 if (Interval > requestTimeOut) {98 logger.info("====请求超时====");99 return flag;
100 }
101 if(!isLockedInRedis(key,keyTimeout)){
102 logger.info("====轮询结束,添加同步锁====");
103 flag = false;
104 keyList.add(key);
105 break;
106 }
107 }
108 }else{
109 //不等待,直接返回
110 logger.info("====产品已被占用,直接返回====");
111 return flag;
112 }
113
114 }else{
115 logger.info("====产品未被占用,添加同步锁====");
116 flag = false;
117 keyList.add(key);
118 }
119 return flag;
120 }
121
122 /**
123 * 在缓存中查询key是否存在
124 * 若存在则返回true;
125 * 若不存在则将key放入缓存,设置过期时间,返回false
126 * @param key
127 * @param keyTimeout key超时时间单位是秒
128 * @return
129 */
130 boolean isLockedInRedis(String key,int keyTimeout){
131 logger.info("====在缓存中查询key是否存在====");
132 boolean isExist = false;
133 //与redis交互,查询对象是否上锁
134 Long result = this.redisClientTemplate.setnx(key, uuid);
135 logger.info("====上锁 result = "+result+"====");
136 if(null != result && 1 == Integer.parseInt(result.toString())){
137 logger.info("====设置缓存失效时长 = "+keyTimeout+"秒====");
138 this.redisClientTemplate.expire(key, keyTimeout);
139 logger.info("====上锁成功====");
140 isExist = false;
141 }else{
142 logger.info("====上锁失败====");
143 isExist = true;
144 }
145 return isExist;
146 }
147
148 /**
149 * 根据传入key,对该产品进行解锁
150 * @param key
151 * @return
152 */
153 public void unlock(){
154 //与redis交互,对产品解锁
155 if(keyList.size()>0){
156 for(String key : this.keyList){
157 String value = this.redisClientTemplate.get(key);
158 if(null != value && !"".equals(value)){
159 if(uuid.equals(value)){
160 logger.info("====解锁key:"+key+" value="+value+"====");
161 this.redisClientTemplate.del(key);
162 }else{
163 logger.info("====待解锁集合中key:"+key+" value="+value+"与uuid不匹配====");
164 }
165 }else{
166 logger.info("====待解锁集合中key="+key+"的value为空====");
167 }
168 }
169 }else{
170 logger.info("====待解锁集合为空====");
171 }
172 }
173
174
175 }
2.2、业务调用模拟样例
1 //获取同步锁工具类2 SynchrolockUtil synchrolockUtil = SpringUtils.getBean("synchrolockUtil");3 //获取需上锁资源的KEY4 String key = "abc";5 //查询是否上锁,上锁轮询,未上锁加锁6 boolean isLocked = synchrolockUtil.islocked(key,synchrolockUtil.RETRYTYPE_WAIT);7 //判断上锁结果8 if(isLocked){9 logger.error("同步锁请求超时并返回 key ="+key);
10 }else{
11 logger.info("====同步锁加锁陈功====");
12 }
13
14 try {
15
16 //执行业务处理
17
18 } catch (Exception e) {
19 logger.error("业务异常:"+e.getMessage(), e);
20 }finally{
21 //解锁
22 synchrolockUtil.unlock();
23 }
2.3、如果业务处理内部,还有嵌套加锁需求,只需将对象传入方法内部,加锁成功后将key值追加到集合中即可
ps:实际实现中还需要jedis工具类,需额外添加调用
Redis 实现同步锁相关推荐
- SpringBoot整合Redis使用同步锁
如果项目已经用到缓存技术了,一般访问量都比较多了吧,刚才我们上面的请求只是做了缓存处理,考虑一个问题:当在高并发条件下,我们假设至少有1W人同时访问获取学生的信息. List<Student&g ...
- redis实现同步锁的一种实例
先定义lock对象 package com.huazhu.erp.commonbc.domain.lock;import com.huazhu.base.utils.DateUtil; import ...
- java redis的同步_java同步系列之redis分布式锁进化史
标题: 死磕 java同步系列之redis分布式锁进化史 - 彤哥读源码 - 博客园 转帖原地址: https://www.cnblogs.com/tong-yuan/p/11621361.html ...
- 分布式锁的三种实现方式_基于 redis 的分布式锁实现
云龙 资深运维开发工程师,负责游戏系统配置管理平台的设计和开发,目前专注于新 CMDB 系统的开发,平时也关注运维自动化,devops,python 开发等技术. 背景 CMDB 系统里面的机器数据会 ...
- 基于 Redis 实现分布式锁思考
以下文章来源方志朋的博客,回复"666"获面试宝典 来源:blog.csdn.net/xuan_lu/article/details/111600302 分布式锁 基于redis实 ...
- nx set 怎么实现的原子性_基于Redis的分布式锁实现
前言 本篇文章主要介绍基于Redis的分布式锁实现到底是怎么一回事,其中参考了许多大佬写的文章,算是对分布式锁做一个总结 分布式锁概览 在多线程的环境下,为了保证一个代码块在同一时间只能由一个线程访问 ...
- Zookeeper和Redis实现分布式锁,附我的可靠性分析
作者:今天你敲代码了吗 链接:https://www.jianshu.com/p/b6953745e341 在分布式系统中,为保证同一时间只有一个客户端可以对共享资源进行操作,需要对共享资源加锁来实现 ...
- 基于Redis的分布式锁和Redlock算法
来自:后端技术指南针 1 前言 今天开始来和大家一起学习一下Redis实际应用篇,会写几个Redis的常见应用. 在我看来Redis最为典型的应用就是作为分布式缓存系统,其他的一些应用本质上并不是杀手 ...
- 《Redis官方文档》用Redis构建分布式锁
<Redis官方文档>用Redis构建分布式锁 用Redis构建分布式锁 在不同进程需要互斥地访问共享资源时,分布式锁是一种非常有用的技术手段. 有很多三方库和文章描述如何用Redis实现 ...
最新文章
- HashMap 在并发下可能出现的问题分析!
- 深度学习技术在美图个性化推荐的应用实践
- python3项目-把项目从Python2.x移植到Python3.x的经验总结
- 我的zencart 安装流程
- python基础知识05-控制流程
- 三极管工作原理_4种集电极-基极负反馈式三极管偏置电路的工作原理分析
- nodejs中path的用法
- 图解 Python 算法
- wordpress插件翻译不生效_谷歌浏览器自带翻译功能如何设置?不喜欢用插件的快来看...
- 春节档电影评分出炉:韩寒《四海》垫底仅有 5.6 分,他居然第一
- mysql主从北_mysql北非之主从同步
- 为什么要用maven - 1
- mysql死锁 简单例子_写一个Mysql死锁的例子
- 手把手教你写电商爬虫-第四课 淘宝网商品爬虫自动JS渲染
- linux停止license服务器,LICENSE · 机器不学习/linux-command - Gitee.com
- 手把手教你使用Python做数据分析
- 职称计算机考试题库破解版2017,2017职称计算机考试题库及答案
- Mac下好用的软件推荐系统设置
- 怎么拯救一个不大靠谱的数据库系统 (5 不靠谱的前生)
- 测试理论-代码的检查,走查(三)
热门文章
- Outlook 解决“附件大小超出了允许的范围”
- 希尔顿旗下多家品牌酒店重返拉斯维加斯,通过快速扩张加倍押注赌城未来
- android webview中加入无线淘宝客页面点击宝贝详情链接进入宝贝详情页面后无法返回问题
- easyui combotree 使用
- 网页渲染markdown(markdown-it)
- Notepad++强大的代码补全和代码提示功能的方法
- Java和matlab混合编程 java调用.jar包中方法时参数传递出错的解决办法
- CentOS7配置jdk环境变量
- 画册设计如何紧随当代的潮流
- 输出一个由*组成的三角形图案_剪纸手工教程:分享漂亮的手帕剪纸,图案简单又好剪...