欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時(shí)間:2021年01月25日 10:15:29   作者:MIC2016  
這篇文章主要介紹了Redis 實(shí)現(xiàn)同步鎖案例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧

1、技術(shù)方案

1.1、redis的基本命令

1)SETNX命令(SET if Not eXists)

語(yǔ)法:SETNX key value

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

2)expire命令

語(yǔ)法:expire KEY seconds

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

3)DEL命令

語(yǔ)法:DEL key [KEY …]

功能:刪除給定的一個(gè)或多個(gè) key ,不存在的 key 會(huì)被忽略。

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

1)加鎖:“鎖”就是一個(gè)存儲(chǔ)在redis里的key-value對(duì),key是把一組投資操作用字符串來形成唯一標(biāo)識(shí),value其實(shí)并不重要,因?yàn)橹灰@個(gè)唯一的key-value存在,就表示這個(gè)操作已經(jīng)上鎖。

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

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

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

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

1.3、處理流程圖  

2、代碼實(shí)現(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)對(duì)象已加鎖時(shí),設(shè)置為等待并輪詢
 public final String RETRYTYPE_NOWAIT = "0";  //加鎖方法當(dāng)對(duì)象已加鎖時(shí),設(shè)置為直接返回
 private String requestTimeOutName = "";  //投資同步鎖請(qǐng)求超時(shí)時(shí)間
 private String retryIntervalName = "";   //投資同步鎖輪詢間隔
 private String keyTimeoutName = "";  //緩存中key的失效時(shí)間
 private String investProductSn = "";   //產(chǎn)品Sn
 private String uuid;    //對(duì)象唯一標(biāo)識(shí)
 private Long startTime = System.currentTimeMillis(); //首次調(diào)用時(shí)間
 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();
 }
 /**
  * 根據(jù)傳入key值,判斷緩存中是否存在該key
  * 存在-已上鎖:判斷retryType,輪詢超時(shí),或直接返回,返回ture
  * 不存在-未上鎖:將該放入緩存,返回false
  * @param key
  * @param retryType 當(dāng)遇到上鎖情況時(shí) 1:輪詢;0:直接返回
  * @return
  */
 public boolean islocked(String key,String retryType){
  boolean flag = true;
  logger.info("====投資同步鎖設(shè)置輪詢間隔、請(qǐng)求超時(shí)時(shí)長(zhǎng)、緩存key失效時(shí)長(zhǎng)====");
  //投資同步鎖輪詢間隔 毫秒
  Long retryInterval = Long.parseLong(Constants.getProperty(retryIntervalName));
  //投資同步鎖請(qǐng)求超時(shí)時(shí)間 毫秒
  Long requestTimeOut = Long.parseLong(Constants.getProperty(requestTimeOutName));
  //緩存中key的失效時(shí)間 秒
  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("====判斷請(qǐng)求是否超時(shí)====");
     Long currentTime = System.currentTimeMillis(); //當(dāng)前調(diào)用時(shí)間
     long Interval = currentTime - startTime;
     if (Interval > requestTimeOut) {
      logger.info("====請(qǐng)求超時(shí)====");
      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è)置過期時(shí)間,返回false
  * @param key
  * @param keyTimeout key超時(shí)時(shí)間單位是秒
  * @return
  */
 boolean isLockedInRedis(String key,int keyTimeout){
  logger.info("====在緩存中查詢key是否存在====");
  boolean isExist = false;
  //與redis交互,查詢對(duì)象是否上鎖
  Long result = this.redisClientTemplate.setnx(key, uuid);
  logger.info("====上鎖 result = "+result+"====");
  if(null != result && 1 == Integer.parseInt(result.toString())){
   logger.info("====設(shè)置緩存失效時(shí)長(zhǎng) = "+keyTimeout+"秒====");
   this.redisClientTemplate.expire(key, keyTimeout);
   logger.info("====上鎖成功====");
   isExist = false;
  }else{
   logger.info("====上鎖失敗====");
   isExist = true;
  }
  return isExist;
 }
 /**
  * 根據(jù)傳入key,對(duì)該產(chǎn)品進(jìn)行解鎖
  * @param key
  * @return
  */
 public void unlock(){
  //與redis交互,對(duì)產(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("同步鎖請(qǐng)求超時(shí)并返回 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)部,還有嵌套加鎖需求,只需將對(duì)象傳入方法內(nèi)部,加鎖成功后將key值追加到集合中即可

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

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

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

解決方法

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

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

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

相關(guān)文章

  • Redis命令使用技巧之Keys的相關(guān)操作

    Redis命令使用技巧之Keys的相關(guān)操作

    Redis KEYS命令用于搜索具有匹配模式的鍵。下面這篇文章主要給大家介紹了關(guān)于Redis命令使用技巧之Keys的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-10-10
  • RedisAPI原子性操作及原理解析

    RedisAPI原子性操作及原理解析

    這篇文章主要介紹了RedisAPI原子性操作及原理解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-12-12
  • 深入理解Redis7哨兵模式(保姆級(jí)教學(xué))

    深入理解Redis7哨兵模式(保姆級(jí)教學(xué))

    Redis的主從復(fù)制存在一定的缺陷,為了解決這一問題,Redis官方推薦一種高可用方案哨兵模式,本文主要介紹了深入理解Redis7哨兵模式,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-01-01
  • redis2.8配置文件中文翻譯版

    redis2.8配置文件中文翻譯版

    這篇文章主要介紹了redis2.8配置文件中文翻譯版,本文翻譯了配置文件中的參數(shù)說明,非常詳細(xì),需要的朋友可以參考下
    2015-06-06
  • redisson滑動(dòng)時(shí)間窗應(yīng)用場(chǎng)景解決方案

    redisson滑動(dòng)時(shí)間窗應(yīng)用場(chǎng)景解決方案

    前10分鐘內(nèi)累計(jì)3次驗(yàn)證失敗后,增加圖形驗(yàn)證碼驗(yàn)證條件,前10分鐘內(nèi)累計(jì)6次驗(yàn)證失敗后,系統(tǒng)自動(dòng)鎖定該賬號(hào)15分鐘,15分鐘后自動(dòng)解鎖,本文給大家分享redisson滑動(dòng)時(shí)間窗應(yīng)用場(chǎng)景解決方案,感興趣的朋友一起看看吧
    2024-01-01
  • redis快速部署為docker容器的方法實(shí)現(xiàn)

    redis快速部署為docker容器的方法實(shí)現(xiàn)

    部署 Redis 作為 Docker 容器是一種快速、靈活且可重復(fù)使用的方式,特別適合開發(fā)、測(cè)試和部署環(huán)境,本文主要介紹了redis快速部署為docker容器的方法實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-05-05
  • Redis中的3種特殊數(shù)據(jù)結(jié)構(gòu)詳解

    Redis中的3種特殊數(shù)據(jù)結(jié)構(gòu)詳解

    在本文中,我們對(duì)三種特殊的數(shù)據(jù)類型進(jìn)行了介紹,它們分別是geospatial(地理空間數(shù)據(jù)類型)、HyperLogLogs和Bitmaps(位圖),這些數(shù)據(jù)類型在不同的領(lǐng)域和應(yīng)用中發(fā)揮著重要作用,并且具有各自獨(dú)特的特性和用途,對(duì)Redis特殊數(shù)據(jù)結(jié)構(gòu)相關(guān)知識(shí)感興趣的朋友一起看看吧
    2024-02-02
  • Redis簡(jiǎn)介

    Redis簡(jiǎn)介

    Redis是一個(gè)開源,高級(jí)的鍵值存儲(chǔ)和一個(gè)適用的解決方案,用于構(gòu)建高性能,可擴(kuò)展的Web應(yīng)用程序。關(guān)于redis的相關(guān)知識(shí)大家可以通過本教程學(xué)習(xí)
    2017-05-05
  • Redis中一些最常見的面試問題總結(jié)

    Redis中一些最常見的面試問題總結(jié)

    Redis在互聯(lián)網(wǎng)技術(shù)存儲(chǔ)方面使用如此廣泛,幾乎所有的后端技術(shù)面試官都要在Redis的使用和原理方面對(duì)小伙伴們進(jìn)行各種刁難。下面這篇文章主要給大家總結(jié)介紹了關(guān)于Redis中一些最常見的面試問題,需要的朋友可以參考下
    2018-09-09
  • Redis 布隆過濾器的原理和實(shí)踐教程

    Redis 布隆過濾器的原理和實(shí)踐教程

    布隆過濾器適用于需要快速判斷一個(gè)元素是否可能存在于集合中的場(chǎng)景,例如網(wǎng)絡(luò)爬蟲中的去重、緩存中的數(shù)據(jù)判斷等,這篇文章主要介紹了Redis 布隆過濾器的原理和實(shí)踐,需要的朋友可以參考下
    2024-02-02

最新評(píng)論