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、同步锁工具类

 package com.mic.synchrolock.util;

 import java.util.ArrayList;
import java.util.List;
import java.util.UUID; import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory; import org.springframework.beans.factory.annotation.Autowired; import com.mic.constants.Constants;
import com.mic.constants.InvestType; /**
* 分布式同步锁工具类
* @author Administrator
*
*/
public class SynchrolockUtil { private final Log logger = LogFactory.getLog(getClass()); @Autowired
private RedisClientTemplate redisClientTemplate; public final String RETRYTYPE_WAIT = "1"; //加锁方法当对象已加锁时,设置为等待并轮询
public final String RETRYTYPE_NOWAIT = "0"; //加锁方法当对象已加锁时,设置为直接返回 private String requestTimeOutName = ""; //投资同步锁请求超时时间
private String retryIntervalName = ""; //投资同步锁轮询间隔
private String keyTimeoutName = ""; //缓存中key的失效时间
private String investProductSn = ""; //产品Sn
private String uuid; //对象唯一标识 private Long startTime = System.currentTimeMillis(); //首次调用时间
public Long getStartTime() {
return startTime;
} List<String> keyList = new ArrayList<String>(); //缓存key的保存集合
public List<String> getKeyList() {
return keyList;
}
public void setKeyList(List<String> keyList) {
this.keyList = keyList;
} @PostConstruct
public void init() {
uuid = UUID.randomUUID().toString();
} @PreDestroy
public void destroy() {
this.unlock();
} /**
* 根据传入key值,判断缓存中是否存在该key
* 存在-已上锁:判断retryType,轮询超时,或直接返回,返回ture
* 不存在-未上锁:将该放入缓存,返回false
* @param key
* @param retryType 当遇到上锁情况时 1:轮询;0:直接返回
* @return
*/
public boolean islocked(String key,String retryType){
boolean flag = true;
logger.info("====投资同步锁设置轮询间隔、请求超时时长、缓存key失效时长====");
//投资同步锁轮询间隔 毫秒
Long retryInterval = Long.parseLong(Constants.getProperty(retryIntervalName));
//投资同步锁请求超时时间 毫秒
Long requestTimeOut = Long.parseLong(Constants.getProperty(requestTimeOutName));
//缓存中key的失效时间 秒
Integer keyTimeout = Integer.parseInt(Constants.getProperty(keyTimeoutName)); //调用缓存获取当前产品锁
logger.info("====当前产品key为:"+key+"====");
if(isLockedInRedis(key,keyTimeout)){
if("1".equals(retryType)){
//采用轮询方式等待
while (true) {
logger.info("====产品已被占用,开始轮询====");
try {
Thread.sleep(retryInterval);
} catch (InterruptedException e) {
logger.error("线程睡眠异常:"+e.getMessage(), e);
return flag;
}
logger.info("====判断请求是否超时====");
Long currentTime = System.currentTimeMillis(); //当前调用时间
long Interval = currentTime - startTime;
if (Interval > requestTimeOut) {
logger.info("====请求超时====");
return flag;
}
if(!isLockedInRedis(key,keyTimeout)){
logger.info("====轮询结束,添加同步锁====");
flag = false;
keyList.add(key);
break;
}
}
}else{
//不等待,直接返回
logger.info("====产品已被占用,直接返回====");
return flag;
} }else{
logger.info("====产品未被占用,添加同步锁====");
flag = false;
keyList.add(key);
}
return flag;
} /**
* 在缓存中查询key是否存在
* 若存在则返回true;
* 若不存在则将key放入缓存,设置过期时间,返回false
* @param key
* @param keyTimeout key超时时间单位是秒
* @return
*/
boolean isLockedInRedis(String key,int keyTimeout){
logger.info("====在缓存中查询key是否存在====");
boolean isExist = false;
//与redis交互,查询对象是否上锁
Long result = this.redisClientTemplate.setnx(key, uuid);
logger.info("====上锁 result = "+result+"====");
if(null != result && 1 == Integer.parseInt(result.toString())){
logger.info("====设置缓存失效时长 = "+keyTimeout+"秒====");
this.redisClientTemplate.expire(key, keyTimeout);
logger.info("====上锁成功====");
isExist = false;
}else{
logger.info("====上锁失败====");
isExist = true;
}
return isExist;
} /**
* 根据传入key,对该产品进行解锁
* @param key
* @return
*/
public void unlock(){
//与redis交互,对产品解锁
if(keyList.size()>0){
for(String key : this.keyList){
String value = this.redisClientTemplate.get(key);
if(null != value && !"".equals(value)){
if(uuid.equals(value)){
logger.info("====解锁key:"+key+" value="+value+"====");
this.redisClientTemplate.del(key);
}else{
logger.info("====待解锁集合中key:"+key+" value="+value+"与uuid不匹配====");
}
}else{
logger.info("====待解锁集合中key="+key+"的value为空====");
}
}
}else{
logger.info("====待解锁集合为空====");
}
} }

2.2、业务调用模拟样例

   //获取同步锁工具类
  SynchrolockUtil synchrolockUtil = SpringUtils.getBean("synchrolockUtil");
  //获取需上锁资源的KEY
  String key = "abc";
  //查询是否上锁,上锁轮询,未上锁加锁
  boolean isLocked = synchrolockUtil.islocked(key,synchrolockUtil.RETRYTYPE_WAIT);
  //判断上锁结果
  if(isLocked){
  logger.error("同步锁请求超时并返回 key ="+key);
  }else{
  logger.info("====同步锁加锁陈功====");
  }   try {   //执行业务处理   } catch (Exception e) {
  logger.error("业务异常:"+e.getMessage(), e);
  }finally{
  //解锁
   synchrolockUtil.unlock();
  }

2.3、如果业务处理内部,还有嵌套加锁需求,只需将对象传入方法内部,加锁成功后将key值追加到集合中即可

ps:实际实现中还需要jedis工具类,需额外添加调用

结合 Redis 实现同步锁的更多相关文章

  1. 基于redis 实现分布式锁的方案

    在电商项目中,经常有秒杀这样的活动促销,在并发访问下,很容易出现上述问题.如果在库存操作上,加锁就可以避免库存卖超的问题.分布式锁使分布式系统之间同步访问共享资源的一种方式 基于redis实现分布式锁 ...

  2. 用Redis构建分布式锁-RedLock(真分布)

    在不同进程需要互斥地访问共享资源时,分布式锁是一种非常有用的技术手段. 有很多三方库和文章描述如何用Redis实现一个分布式锁管理器,但是这些库实现的方式差别很大,而且很多简单的实现其实只需采用稍微增 ...

  3. Redis实现分布式锁

    http://redis.io/topics/distlock 在不同进程需要互斥地访问共享资源时,分布式锁是一种非常有用的技术手段. 有很多三方库和文章描述如何用Redis实现一个分布式锁管理器,但 ...

  4. 基于Redis实现分布式锁(1)

    转自:http://blog.csdn.net/ugg/article/details/41894947 背景在很多互联网产品应用中,有些场景需要加锁处理,比如:秒杀,全局递增ID,楼层生成等等.大部 ...

  5. redis咋么实现分布式锁,redis分布式锁的实现方式,redis做分布式锁 积极正义的少年

    前言 分布式锁一般有三种实现方式:1. 数据库乐观锁:2. 基于Redis的分布式锁:3. 基于ZooKeeper的分布式锁.本篇博客将介绍第二种方式,基于Redis实现分布式锁.虽然网上已经有各种介 ...

  6. Redis实现分布式锁的正确姿势

    分布式锁一般有三种实现方式:1. 数据库乐观锁:2. 基于Redis的分布式锁:3. 基于ZooKeeper的分布式锁.本篇博客将介绍第二种方式,基于Redis实现分布式锁.虽然网上已经有各种介绍Re ...

  7. Java中String做为synchronized同步锁使用详解

    Java中使用String作同步锁 在Java中String是一种特殊的类型存在,在jdk中String在创建后是共享常量池的,即使在jdk1.8之后实现有所不同,但是功能还是差不多的. 借助这个特点 ...

  8. 如何优雅地用Redis实现分布式锁?

    转: 如何优雅地用Redis实现分布式锁?   BaiduSpring 01-2500:01 什么是分布式锁 在学习Java多线程编程的时候,锁是一个很重要也很基础的概念,锁可以看成是多线程情况下访问 ...

  9. 基于Redis的分布式锁真的安全吗?

    说明: 我前段时间写了一篇用consul实现分布式锁,感觉理解的也不是很好,直到我看到了这2篇写分布式锁的讨论,真的是很佩服作者严谨的态度, 把这种分布式锁研究的这么透彻,作者这种技术态度真的值得我好 ...

随机推荐

  1. Docker概念学习系列之Docker核心概念之镜像Image

    不多说,直接上干货! 说明:   Docker 运行容器之前需要本地存在对应的镜像,如果镜像不存在,Docker 会尝试先从默认镜像仓库下载(默认使用Docker Hub公共注册服务器中的仓库),用户 ...

  2. 【JAVA】异常笔记

    自定义异常需要注意: 所有异常都必须是 Throwable 的子类. 如果希望写一个检查性异常类,则需要继承 Exception 类. 如果你想写一个运行时异常类,那么需要继承 RuntimeExce ...

  3. Spring Boot + Spring Cloud 实现权限管理系统 后端篇(七):集成 Druid 数据源

    数据库连接池负责分配.管理和释放数据库连接,它允许应用程序重复使用一个现有的数据库连接,而不是再重新建立一个:释放空闲时间超过最大空闲时间的数据库连接来避免因为没有释放数据库连接而引起的数据库连接遗漏 ...

  4. tensorflow 根据节点获取节点前的整张图

    1.先获取节点 output_layer = self.model.get_pooled_output() logits = self.tf_instance.matmul(output_layer, ...

  5. springboot 入门

    使用maven构建project项目, 配置aliyun仓库, 不赘述 springboot 版本需要: jdk1.7+, maven3.2+ , gradle2.9+ 配置文件 引入父包, 放在&l ...

  6. 查看 postgresql 数据库编码,以及修改数据库编码

    查看数据表编码: \encoding 修改数据库编码: update pg_database set encoding = pg_char_to_encoding('UTF8') where datn ...

  7. AI---训练集(train set) 验证集(validation set) 测试集(test set)

    在有监督(supervise)的机器学习中,数据集常被分成2~3个即: 训练集(train set) 验证集(validation set) 测试集(test set) 一般需要将样本分成独立的三部分 ...

  8. spring-boot-mail

    1. 功能 发送普通邮件 发送htm邮件 发送带附件的邮件 发送带静态资源的邮件 2. 实现 类结构图 3. 实现 接口 package com.jihite.service; public inte ...

  9. APP---发布动态、朋友圈类似,多张图片动响应式正方形展示布局 vue.js,aui.css,apiclouv

    环境:vue.js,aui.css,apicloud 1.没做控制之前.图片真实长度宽度. 2.下面用js控制高度 js部分 //js 部分 //先动态的获取属性宽度 var box4_col3 = ...

  10. [转](SQL Server) Convert a File from utf-8 to ANSI (such as Windows-1252)

    本文转自:https://example-code.com/sql/charset_convert_file_from_utf8_to_ansi.asp CREATE PROCEDURE Chilka ...