亚洲综合原千岁中文字幕_国产精品99久久久久久久vr_无码人妻aⅴ一区二区三区浪潮_成人h动漫精品一区二区三

主頁 > 知識庫 > Redis 實現同步鎖案例

Redis 實現同步鎖案例

熱門標簽:超呼電話機器人 山東外呼銷售系統招商 貴州電銷卡外呼系統 宿遷便宜外呼系統平臺 鄭州人工智能電銷機器人系統 魔獸2青云地圖標注 北京400電話辦理收費標準 日本中國地圖標注 十堰營銷電銷機器人哪家便宜

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;
 }
 ListString> keyList = new ArrayListString>(); //緩存key的保存集合
 public ListString> getKeyList() {
  return keyList;
 }
 public void setKeyList(ListString> 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鎖還是出現同步問題

一種可能是,2臺機器同時訪問,一臺訪問,還沒有把鎖設置過去的時候,另一臺也查不到就會出現這個問題。

解決方法

這我跟寫代碼的方式有關。先查,如果不存在就set,這種方式有極微小的可能存在時間差,導致鎖set了2次。

推薦使用setIfAbsent 這樣在redis set的時候是單線程的。不會存在重復的問題。

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

您可能感興趣的文章:
  • Redis的主從同步解析
  • 簡單注解實現集群同步鎖(spring+redis+注解)
  • SpringBoot集成redis實現分布式鎖的示例代碼
  • 基于redis setIfAbsent的使用說明
  • Redis實現分布式Session管理的機制詳解
  • kubernetes環境部署單節點redis數據庫的方法

標簽:果洛 臺州 朝陽 江蘇 大慶 北京 楊凌 吉安

巨人網絡通訊聲明:本文標題《Redis 實現同步鎖案例》,本文關鍵詞  Redis,實現,同步,鎖,案例,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Redis 實現同步鎖案例》相關的同類信息!
  • 本頁收集關于Redis 實現同步鎖案例的相關信息資訊供網民參考!
  • 推薦文章
    欧美大片a一级毛片视频| 精品视频一区二区三区免费| 久草免费在线观看| 青草国产在线| 99久久网站| 精品久久久久久中文字幕一区| 欧美a免费| 九九久久99| 四虎影视久久久| 99热热久久| 日韩在线观看免费完整版视频| 999精品视频在线| 久久精品欧美一区二区| 国产网站免费观看| 免费的黄视频| 九九精品在线| 国产一区二区福利久久| 欧美激情一区二区三区视频高清| 国产成人精品综合久久久| 黄色免费网站在线| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 亚洲精品永久一区| 精品国产一区二区三区久久久狼| 黄视频网站免费| 日本久久久久久久 97久久精品一区二区三区 狠狠色噜噜狠狠狠狠97 日日干综合 五月天婷婷在线观看高清 九色福利视频 | 精品在线观看一区| 天堂网中文在线| 成人免费一级毛片在线播放视频| 亚洲精品中文一区不卡| 国产伦久视频免费观看 视频| 韩国三级一区| 999久久狠狠免费精品| 国产欧美精品午夜在线播放| 亚飞与亚基在线观看| 欧美日本免费| 国产一区二区精品在线观看| 麻豆污视频| 四虎久久影院| 一级女人毛片人一女人| 青青青草视频在线观看| 99久久精品国产免费| 国产韩国精品一区二区三区| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 欧美国产日韩一区二区三区| 精品视频在线看| 国产一区二区福利久久| 久久精品免视看国产明星| 久久99欧美| 欧美大片一区| 成人高清视频免费观看| 日韩在线观看视频黄| 成人免费一级毛片在线播放视频| 久久精品免视看国产明星| 黄色福利| 天天做日日爱| 成人在免费观看视频国产| 黄视频网站免费| 国产精品免费久久| 精品国产一区二区三区精东影业| 精品久久久久久中文| 精品视频在线看| 毛片成人永久免费视频| 日日日夜夜操| 黄色福利| 欧美另类videosbestsex视频| 亚欧成人乱码一区二区| 国产网站在线| 精品国产一级毛片| 国产福利免费视频| 国产成a人片在线观看视频| 99久久精品费精品国产一区二区| 午夜欧美成人香蕉剧场| 色综合久久手机在线| 久久99欧美| 天天做日日爱夜夜爽| 九九干| 亚洲 欧美 91| 国产一区二区福利久久| 精品久久久久久中文字幕2017| 99久久精品国产免费| 欧美α片无限看在线观看免费| 精品视频免费在线| 一级女性全黄久久生活片| 国产a毛片| 可以在线看黄的网站| 99久久网站| 成人影视在线观看| 国产综合成人观看在线| a级毛片免费观看网站| 四虎影视精品永久免费网站 | 日韩综合| 日本特黄特色aaa大片免费| 成人免费福利片在线观看| 久久成人综合网| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 日韩在线观看免费| 韩国三级一区| 久久国产影院| 国产一区免费在线观看| 国产视频久久久| 韩国三级视频网站| 日本特黄特黄aaaaa大片| 国产一区免费在线观看| 九九久久国产精品大片| 韩国毛片| 二级片在线观看| 日韩中文字幕在线播放| 精品国产一区二区三区国产馆| 久久国产影视免费精品| 美女被草网站| 欧美α片无限看在线观看免费| 精品视频在线观看一区二区三区| 亚洲精品中文字幕久久久久久| 亚洲 男人 天堂| 九九久久国产精品大片| 麻豆系列 在线视频| 青青青草视频在线观看| 日韩中文字幕一区| 午夜久久网| 欧美电影免费| 四虎精品在线观看| 九九精品影院| 日韩在线观看视频免费| 久久久久久久男人的天堂| 亚洲爆爽| 国产极品白嫩美女在线观看看| 欧美一级视频免费观看| 国产一区免费在线观看| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 欧美爱爱网| 欧美大片a一级毛片视频| 国产高清在线精品一区a| 国产a视频| 国产综合成人观看在线| 久久精品店| 国产成人啪精品| 欧美1区| 四虎影视久久| 欧美a免费| 久久国产精品永久免费网站| 国产成人精品综合久久久| 韩国毛片基地| 亚洲天堂免费| 国产一区国产二区国产三区| 国产不卡精品一区二区三区| 久久99欧美| 亚洲精品影院| 日本免费乱理伦片在线观看2018| 韩国毛片免费| 欧美大片毛片aaa免费看| 国产不卡在线看| 国产91精品一区二区| 久草免费在线视频| 亚洲天堂在线播放| 九九免费精品视频| 精品国产一区二区三区精东影业 | 成人高清护士在线播放| 欧美激情中文字幕一区二区| 韩国三级视频网站| 天天做日日爱夜夜爽| 免费一级生活片| 精品视频在线看| 九九精品在线| 国产欧美精品午夜在线播放| 日韩av东京社区男人的天堂| 黄色免费三级| 国产高清在线精品一区二区| 日日夜人人澡人人澡人人看免| 久久成人综合网| 99久久精品国产高清一区二区| 一级毛片看真人在线视频| 欧美1区| 成人免费观看视频| 91麻豆精品国产高清在线| 999久久狠狠免费精品| 国产麻豆精品免费视频| 成人免费一级毛片在线播放视频| 日韩中文字幕在线亚洲一区 | 天堂网中文在线| 91麻豆爱豆果冻天美星空| 日韩专区在线播放| 香蕉视频一级| 欧美大片毛片aaa免费看| 日韩欧美一二三区| 欧美另类videosbestsex久久| 天堂网中文字幕| 成人av在线播放| 四虎精品在线观看| 国产高清在线精品一区二区| 国产一区二区精品久久| 日韩av东京社区男人的天堂| 国产成人精品综合| 日本免费看视频| 中文字幕一区二区三区精彩视频| 国产麻豆精品hdvideoss| 国产不卡高清在线观看视频| 韩国三级视频在线观看| a级毛片免费观看网站| 超级乱淫黄漫画免费| 日本伦理黄色大片在线观看网站|