婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Redis 實現(xiàn)同步鎖案例

Redis 實現(xiàn)同步鎖案例

熱門標(biāo)簽:超呼電話機(jī)器人 山東外呼銷售系統(tǒng)招商 貴州電銷卡外呼系統(tǒng) 宿遷便宜外呼系統(tǒng)平臺 鄭州人工智能電銷機(jī)器人系統(tǒng) 魔獸2青云地圖標(biāo)注 北京400電話辦理收費(fèi)標(biāo)準(zhǔn) 日本中國地圖標(biāo)注 十堰營銷電銷機(jī)器人哪家便宜

1、技術(shù)方案

1.1、redis的基本命令

1)SETNX命令(SET if Not eXists)

語法:SETNX key value

功能:當(dāng)且僅當(dāng) key 不存在,將 key 的值設(shè)為 value ,并返回1;若給定的 key 已經(jīng)存在,則 SETNX 不做任何動作,并返回0。

2)expire命令

語法:expire KEY seconds

功能:設(shè)置key的過期時間。如果key已過期,將會被自動刪除。

3)DEL命令

語法:DEL key [KEY …]

功能:刪除給定的一個或多個 key ,不存在的 key 會被忽略。

1.2、實現(xiàn)同步鎖原理

1)加鎖:“鎖”就是一個存儲在redis里的key-value對,key是把一組投資操作用字符串來形成唯一標(biāo)識,value其實并不重要,因為只要這個唯一的key-value存在,就表示這個操作已經(jīng)上鎖。

2)解鎖:既然key-value對存在就表示上鎖,那么釋放鎖就自然是在redis里刪除key-value對。

3)阻塞、非阻塞:阻塞式的實現(xiàn),若線程發(fā)現(xiàn)已經(jīng)上鎖,會在特定時間內(nèi)輪詢鎖。非阻塞式的實現(xiàn),若發(fā)現(xiàn)線程已經(jīng)上鎖,則直接返回。

4)處理異常情況:假設(shè)當(dāng)投資操作調(diào)用其他平臺接口出現(xiàn)等待時,自然沒有釋放鎖,這種情況下加入鎖超時機(jī)制,用redis的expire命令為key設(shè)置超時時長,過了超時時間redis就會將這個key自動刪除,即強(qiáng)制釋放鎖

(此步驟需在JAVA內(nèi)部設(shè)置同樣的超時機(jī)制,內(nèi)部超時時長應(yīng)小于或等于redis超時時長)。

1.3、處理流程圖  

2、代碼實現(xiàn)

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";  //加鎖方法當(dāng)對象已加鎖時,設(shè)置為等待并輪詢
 public final String RETRYTYPE_NOWAIT = "0";  //加鎖方法當(dāng)對象已加鎖時,設(shè)置為直接返回
 private String requestTimeOutName = "";  //投資同步鎖請求超時時間
 private String retryIntervalName = "";   //投資同步鎖輪詢間隔
 private String keyTimeoutName = "";  //緩存中key的失效時間
 private String investProductSn = "";   //產(chǎn)品Sn
 private String uuid;    //對象唯一標(biāo)識
 private Long startTime = System.currentTimeMillis(); //首次調(diào)用時間
 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();
 }
 /**
  * 根據(jù)傳入key值,判斷緩存中是否存在該key
  * 存在-已上鎖:判斷retryType,輪詢超時,或直接返回,返回ture
  * 不存在-未上鎖:將該放入緩存,返回false
  * @param key
  * @param retryType 當(dāng)遇到上鎖情況時 1:輪詢;0:直接返回
  * @return
  */
 public boolean islocked(String key,String retryType){
  boolean flag = true;
  logger.info("====投資同步鎖設(shè)置輪詢間隔、請求超時時長、緩存key失效時長====");
  //投資同步鎖輪詢間隔 毫秒
  Long retryInterval = Long.parseLong(Constants.getProperty(retryIntervalName));
  //投資同步鎖請求超時時間 毫秒
  Long requestTimeOut = Long.parseLong(Constants.getProperty(requestTimeOutName));
  //緩存中key的失效時間 秒
  Integer keyTimeout = Integer.parseInt(Constants.getProperty(keyTimeoutName));
  //調(diào)用緩存獲取當(dāng)前產(chǎn)品鎖
  logger.info("====當(dāng)前產(chǎn)品key為:"+key+"====");
  if(isLockedInRedis(key,keyTimeout)){
   if("1".equals(retryType)){
    //采用輪詢方式等待
    while (true) {
     logger.info("====產(chǎn)品已被占用,開始輪詢====");
     try {
      Thread.sleep(retryInterval);
     } catch (InterruptedException e) {
      logger.error("線程睡眠異常:"+e.getMessage(), e);
      return flag;
     }
     logger.info("====判斷請求是否超時====");
     Long currentTime = System.currentTimeMillis(); //當(dāng)前調(diào)用時間
     long Interval = currentTime - startTime;
     if (Interval > requestTimeOut) {
      logger.info("====請求超時====");
      return flag;
     }
     if(!isLockedInRedis(key,keyTimeout)){
      logger.info("====輪詢結(jié)束,添加同步鎖====");
      flag = false;
      keyList.add(key);
      break;
     }
    }
   }else{
    //不等待,直接返回
    logger.info("====產(chǎn)品已被占用,直接返回====");
    return flag;
   }
  }else{
   logger.info("====產(chǎn)品未被占用,添加同步鎖====");
   flag = false;
   keyList.add(key);
  }
  return flag;
 }
 /**
  * 在緩存中查詢key是否存在
  * 若存在則返回true;
  * 若不存在則將key放入緩存,設(shè)置過期時間,返回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("====設(shè)置緩存失效時長 = "+keyTimeout+"秒====");
   this.redisClientTemplate.expire(key, keyTimeout);
   logger.info("====上鎖成功====");
   isExist = false;
  }else{
   logger.info("====上鎖失敗====");
   isExist = true;
  }
  return isExist;
 }
 /**
  * 根據(jù)傳入key,對該產(chǎn)品進(jìn)行解鎖
  * @param key
  * @return
  */
 public void unlock(){
  //與redis交互,對產(chǎn)品解鎖
  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、業(yè)務(wù)調(diào)用模擬樣例

//獲取同步鎖工具類
  SynchrolockUtil synchrolockUtil = SpringUtils.getBean("synchrolockUtil");
  //獲取需上鎖資源的KEY
  String key = "abc";
  //查詢是否上鎖,上鎖輪詢,未上鎖加鎖
  boolean isLocked = synchrolockUtil.islocked(key,synchrolockUtil.RETRYTYPE_WAIT);
  //判斷上鎖結(jié)果
  if(isLocked){
   logger.error("同步鎖請求超時并返回 key ="+key);
  }else{
   logger.info("====同步鎖加鎖陳功====");
  }
  try {
   //執(zhí)行業(yè)務(wù)處理
  } catch (Exception e) {
   logger.error("業(yè)務(wù)異常:"+e.getMessage(), e);
  }finally{
   //解鎖
    synchrolockUtil.unlock();
  }

2.3、如果業(yè)務(wù)處理內(nèi)部,還有嵌套加鎖需求,只需將對象傳入方法內(nèi)部,加鎖成功后將key值追加到集合中即可

ps:實際實現(xiàn)中還需要jedis工具類,需額外添加調(diào)用

補(bǔ)充:使用redis鎖還是出現(xiàn)同步問題

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

解決方法

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

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

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

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

標(biāo)簽:果洛 臺州 朝陽 江蘇 大慶 北京 楊凌 吉安

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Redis 實現(xiàn)同步鎖案例》,本文關(guān)鍵詞  Redis,實現(xiàn),同步,鎖,案例,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Redis 實現(xiàn)同步鎖案例》相關(guān)的同類信息!
  • 本頁收集關(guān)于Redis 實現(xiàn)同步鎖案例的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    国产欧美日韩不卡| 激情偷乱视频一区二区三区| 国产精品福利影院| 一区二区高清免费观看影视大全| 丝袜脚交一区二区| 国产成人在线网站| 欧美高清激情brazzers| 国产色婷婷亚洲99精品小说| 夜夜嗨av一区二区三区中文字幕 | 精品一区二区三区视频| 99久久久精品| 91尤物视频在线观看| 日韩精品中文字幕在线一区| 亚洲一区免费观看| 欧美性色黄大片| 亚洲成人免费视| 欧美男人的天堂一二区| 国产精品美女久久久久aⅴ国产馆| 日韩电影在线一区二区三区| 白白色亚洲国产精品| 久久精品男人天堂av| 国产老妇另类xxxxx| 国产女人aaa级久久久级| 精品综合久久久久久8888| 欧美精品在线观看播放| 免播放器亚洲一区| 亚洲国产wwwccc36天堂| 欧美激情中文字幕一区二区| 国产欧美日韩视频一区二区| www国产亚洲精品久久麻豆| 欧美成人a视频| 91精品欧美综合在线观看最新| 99re成人精品视频| 色偷偷88欧美精品久久久| 成人福利电影精品一区二区在线观看| 日韩国产一区二| 久久噜噜亚洲综合| 精品欧美乱码久久久久久1区2区| 精品国产一区二区三区久久久蜜月| 精品国产1区2区3区| 亚洲欧洲在线观看av| 日韩不卡一区二区| 色天天综合色天天久久| 亚洲欧美偷拍三级| 欧美一区二区三区精品| 日韩电影在线观看电影| 欧美一区二区视频网站| jizz一区二区| 国产精品久99| 成人午夜电影网站| 一区二区三区四区国产精品| 欧美视频一二三区| 国产精品888| 欧美久久久久久久久久| 午夜在线电影亚洲一区| jiyouzz国产精品久久| 色美美综合视频| 亚洲国产精品久久艾草纯爱| 欧美电视剧免费观看| 国产一区二区三区视频在线播放| 日韩欧美综合在线| 亚洲小说欧美激情另类| 91麻豆精品在线观看| 国产日韩欧美综合在线| 婷婷亚洲久悠悠色悠在线播放| 在线观看日韩高清av| 亚洲成va人在线观看| 亚洲美女在线一区| www.在线欧美| 久久婷婷成人综合色| 成人免费视频网站在线观看| 精品动漫一区二区三区在线观看| 国产成人在线视频免费播放| 中文字幕电影一区| 欧洲人成人精品| 国内精品伊人久久久久av影院 | 青椒成人免费视频| 精东粉嫩av免费一区二区三区| 日韩视频一区二区三区| 精品一区二区三区在线播放视频| 久久精品一区蜜桃臀影院| 精品一区精品二区高清| 久久久不卡网国产精品一区| 国产乱淫av一区二区三区| 国产精品蜜臀在线观看| 色狠狠av一区二区三区| 亚洲男人电影天堂| 欧美美女一区二区| 国产成人精品亚洲日本在线桃色| 亚洲欧洲日韩女同| 色8久久精品久久久久久蜜| 日韩中文字幕一区二区三区| 精品少妇一区二区三区日产乱码| 高清成人免费视频| 一区二区在线观看视频 | 日韩中文字幕麻豆| 免费成人av资源网| 免费的成人av| 国内精品不卡在线| 精品一区二区免费在线观看| 激情综合网最新| 成人av在线网站| 欧美人牲a欧美精品| 日韩视频永久免费| 亚洲国产成人私人影院tom| 亚洲激情欧美激情| 韩国v欧美v日本v亚洲v| 欧美自拍偷拍午夜视频| 欧美大黄免费观看| 中文字幕综合网| 国产成人免费xxxxxxxx| 91在线观看一区二区| 国产成人一级电影| 久久av资源网| 男人的j进女人的j一区| 亚洲午夜久久久久久久久电影院| 日韩高清不卡一区二区| 国产精品美女久久久久aⅴ国产馆| 久久亚洲一级片| 国产精品乱人伦| 一个色在线综合| 国产精品一区二区三区99| av男人天堂一区| 欧美蜜桃一区二区三区| 欧美日韩久久不卡| 亚洲激情在线播放| 国产一区二区三区不卡在线观看| 欧美视频一区二| 国产精品免费视频网站| 成人精品小蝌蚪| 亚洲日本va在线观看| 免费欧美日韩国产三级电影| 国产精品18久久久久久久网站| av成人动漫在线观看| 欧美日韩一区中文字幕| wwwwww.欧美系列| 蜜臀av性久久久久蜜臀av麻豆| 亚洲欧美日韩国产一区二区三区| 国产精品免费观看视频| 欧美一区二区成人| 不卡一区在线观看| 在线播放亚洲一区| 亚洲欧洲精品一区二区精品久久久| 天天色天天爱天天射综合| 99在线精品免费| 亚洲三级小视频| 国产亲近乱来精品视频| aaa亚洲精品一二三区| 亚洲不卡一区二区三区| 337p日本欧洲亚洲大胆色噜噜| 欧美区在线观看| 中文字幕欧美一| 色婷婷av一区二区三区gif | 亚洲精品菠萝久久久久久久| 精品国产精品一区二区夜夜嗨| 一区二区三区欧美久久| 91视频免费播放| 国产女人18水真多18精品一级做| 97成人超碰视| 亚洲1区2区3区4区| 国产日产精品一区| 色综合久久久网| 日韩国产一二三区| 久久精品一区二区三区av| 一本一本大道香蕉久在线精品| 午夜精品久久一牛影视| 精品国产不卡一区二区三区| av在线不卡观看免费观看| 亚洲另类一区二区| 欧美三电影在线| 国产传媒一区在线| 日韩中文字幕亚洲一区二区va在线 | 一区二区在线免费| 亚洲综合在线视频| 亚洲视频免费看| 亚洲一区二区三区不卡国产欧美 | 风流少妇一区二区| 国产成人精品亚洲777人妖| 丁香婷婷综合色啪| 亚洲不卡av一区二区三区| 精品国产污网站| 91在线porny国产在线看| 激情小说亚洲一区| 日韩av午夜在线观看| 日韩国产在线一| 亚洲一区二区av在线| 国产女主播视频一区二区| 精品少妇一区二区| 51久久夜色精品国产麻豆| 欧美日韩在线观看一区二区| 粉嫩aⅴ一区二区三区四区 | 日本一区二区不卡视频| 国产欧美一区二区在线| 亚洲日本在线天堂| 久久91精品久久久久久秒播| 麻豆一区二区三区| 免费xxxx性欧美18vr| 国产乱国产乱300精品| 91在线观看一区二区| 日韩精品专区在线影院重磅|