亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 數據庫 > Redis > 正文

Redis實現分布式鎖和等待序列的方法示例

2020-10-28 21:30:04
字體:
來源:轉載
供稿:網友

在集群下,經常會因為同時處理發生資源爭搶和并發問題,但是我們都知道同步鎖 synchronized 、 cas 、 ReentrankLock 這些鎖的作用范圍都是 JVM ,說白了在集群下沒啥用。這時我們就需要能在多臺 JVM 之間決定執行順序的鎖了,現在分布式鎖主要有 redis 、 Zookeeper 實現的,還有數據庫的方式,不過性能太差,也就是需要一個第三方的監管。

背景

最近在做一個消費 Kafka 消息的時候發現,由于線上的消費者過多,經常會遇到,多個機器同時處理一個主鍵類型的數據的情況發生,如果最后是執行更新操作的話,也就是一個更新順序的問題,但是如果恰好都需要插入數據的時候,會出現主鍵重復的問題。這是生產上不被允許的(因為公司有異常監管的機制,扣分啥的),這是就需要個分布式鎖了,斟酌后用了 Redis 的實現方式(因為網上例子多)

分析

redis 實現的分布式鎖,實現原理是 set 方法,因為多個線程同時請求的時候,只有一個線程可以成功并返回結果,還可以設置有效期,來避免死鎖的發生,一切都是這么的完美,不過有個問題,在 set 的時候,會直接返回結果,成功或者失敗,不具有阻塞效果,需要我們自己對失敗的線程進程處理,有兩種方式

  • 丟棄
  • 等待重試 由于我們的系統需要這些數據,那么只能重新嘗試獲取。這里使用 redis 的 List 類型實現等待序列的作用

代碼

直接上代碼 其實直接redis的工具類就可以解決了

package com.testimport redis.clients.jedis.Jedis;import java.util.Collections;import java.util.List;/** * @desc redis隊列實現方式 * @anthor  * @date  **/public class RedisUcUitl {  private static final String LOCK_SUCCESS = "OK";  private static final String SET_IF_NOT_EXIST = "NX";  private static final String SET_WITH_EXPIRE_TIME = "PX";  private static final Long RELEASE_SUCCESS = 1L;  private RedisUcUitl() {  }  /**   * logger   **/  /**   * 存儲redis隊列順序存儲 在隊列首部存入   *   * @param key  字節類型   * @param value 字節類型   */  public static Long lpush(Jedis jedis, final byte[] key, final byte[] value) {    return jedis.lpush(key, value);    }  /**   * 移除列表中最后一個元素 并將改元素添加入另一個列表中 ,當列表為空時 將阻塞連接 直到等待超時   *   * @param srckey   * @param dstkey   * @param timeout 0 表示永不超時   * @return   */  public static byte[] brpoplpush(Jedis jedis,final byte[] srckey, final byte[] dstkey, final int timeout) {    return jedis.brpoplpush(srckey, dstkey, timeout);  }  /**   * 返回制定的key,起始位置的redis數據   * @param redisKey   * @param start   * @param end -1 表示到最后   * @return   */  public static List<byte[]> lrange(Jedis jedis,final byte[] redisKey, final long start, final long end) {        return jedis.lrange(redisKey, start, end);  }  /**   * 刪除key   * @param redisKey   */  public static void delete(Jedis jedis, final byte[] redisKey) {         return jedis.del(redisKey);  }  /**   * 嘗試加鎖   * @param lockKey key名稱   * @param requestId 身份標識   * @param expireTime 過期時間   * @return   */  public static boolean tryGetDistributedLock(Jedis jedis,final String lockKey, final String requestId, final int expireTime) {    String result = jedis.set(lockKey, requestId, SET_IF_NOT_EXIST, SET_WITH_EXPIRE_TIME, expireTime);    return LOCK_SUCCESS.equals(result);  }  /**   * 釋放鎖   * @param lockKey key名稱   * @param requestId 身份標識   * @return   */  public static boolean releaseDistributedLock(Jedis jedis,final String lockKey, final String requestId) {    final String script = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";    jedis.eval(script, Collections.singletonList(lockKey), Collections.singletonList(requestId));    return RELEASE_SUCCESS.equals(result);  }}

業務邏輯主要代碼如下

1.先消耗隊列中的

while(true){  // 消費隊列  try{    // 被放入redis隊列的數據 序列化后的    byte[] bytes = RedisUcUitl.brpoplpush(keyStr.getBytes(UTF_8), dstKeyStr.getBytes(UTF_8), 1);    if(bytes == null || bytes.isEmpty()){      // 隊列中沒數據時退出      break;    }    // 反序列化對象    Map<String, Object> singleMap = (Map<String, Object>) ObjectSerialUtil.bytesToObject(bytes);    // 塞入唯一的值 防止被其他線程誤解鎖    String requestId = UUID.randomUUID().toString();    boolean lockGetFlag = RedisUcUitl.tryGetDistributedLock(keyStr,requestId, 100);    if(lockGetFlag){      // 成功獲取鎖 進行業務處理      //TODO      // 處理完畢釋放鎖       boolean freeLock = RedisUcUitl.releaseDistributedLock(keyStr, requestId);    }else{      // 未能獲得鎖放入等待隊列     RedisUcUitl.lpush(keyStr.getBytes(UTF_8), ObjectSerialUtil.objectToBytes(param));      }      }catch(Exception e){    break;  }  }

2.處理最新接到的數據

同樣是走嘗試獲取鎖,獲取不到放入隊列的流程

一般序列化用 fastJson 之列的就可以了,這里用的是 JDK 自帶的,工具類如下

public class ObjectSerialUtil {  private ObjectSerialUtil() {//    工具類  }  /**   * 將Object對象序列化為byte[]   *   * @param obj 對象   * @return byte數組   * @throws Exception   */  public static byte[] objectToBytes(Object obj) throws IOException {    ByteArrayOutputStream bos = new ByteArrayOutputStream();    ObjectOutputStream oos = new ObjectOutputStream(bos);    oos.writeObject(obj);    byte[] bytes = bos.toByteArray();    bos.close();    oos.close();    return bytes;  }  /**   * 將bytes數組還原為對象   *   * @param bytes   * @return   * @throws Exception   */  public static Object bytesToObject(byte[] bytes) {    try {      ByteArrayInputStream bin = new ByteArrayInputStream(bytes);      ObjectInputStream ois = new ObjectInputStream(bin);      return ois.readObject();    } catch (Exception e) {      throw new BaseException("反序列化出錯!", e);    }  }}

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美性猛交xxxx富婆弯腰| 国产精品成人久久久久| 91香蕉嫩草影院入口| 国产精品久久久久一区二区| 久久久亚洲天堂| 久久好看免费视频| 国产成人精品视频在线| 欧美午夜精品久久久久久浪潮| 成人a免费视频| 国语自产偷拍精品视频偷| 久久天天躁狠狠躁夜夜躁| 成人精品一区二区三区电影免费| 国产日韩欧美夫妻视频在线观看| 日日摸夜夜添一区| 日韩欧美在线视频日韩欧美在线视频| 欧美精品在线视频观看| 国产精品美女无圣光视频| 欧美日韩国产123| 精品成人69xx.xyz| 国产精品极品尤物在线观看| 国产精品小说在线| 国产在线观看不卡| 欧美另类99xxxxx| 欧美色视频日本高清在线观看| 91精品视频专区| 国产精品444| 日韩精品极品视频| 成人网在线免费观看| 欧美日韩另类字幕中文| 久久久成人精品视频| 日本一区二区在线播放| 欧美放荡办公室videos4k| 中文字幕国产精品久久| 欧美日韩久久久久| 中文字幕日韩精品有码视频| 91精品在线国产| 午夜精品一区二区三区视频免费看| 成人精品在线观看| 亚洲国产欧美久久| 日韩中文娱乐网| 欧美激情网站在线观看| 亚洲天天在线日亚洲洲精| 国产精品一区二区性色av| 欧美精品一本久久男人的天堂| 亚洲综合精品伊人久久| 一区二区三区美女xx视频| 91九色视频在线| 在线成人免费网站| 欧美亚洲国产成人精品| 国产精品久久久久久婷婷天堂| 中文欧美在线视频| 91精品国产免费久久久久久| 成人免费网站在线| 日韩欧美在线国产| 精品久久久精品| 欧美日韩亚洲一区二区三区| 欧美一区亚洲一区| 国产精品流白浆视频| 日韩电影免费观看在线| 欧美日本精品在线| 8050国产精品久久久久久| 欧美wwwwww| 中文字幕亚洲在线| 久久精品91久久香蕉加勒比| 一本色道久久88综合日韩精品| 黑人精品xxx一区| xxxxx91麻豆| 北条麻妃一区二区在线观看| 最新国产成人av网站网址麻豆| 亚洲3p在线观看| 日韩免费观看网站| 久久亚洲电影天堂| 久久福利网址导航| 色av吧综合网| 亚洲第一精品电影| 日韩电影在线观看永久视频免费网站| 欧美精品中文字幕一区| 欧美激情国产精品| 欧美大尺度在线观看| 粉嫩老牛aⅴ一区二区三区| 欧美极度另类性三渗透| 精品性高朝久久久久久久| 成人国产精品久久久久久亚洲| 日韩电影免费观看中文字幕| 欧美日韩国产在线播放| 97在线看福利| 68精品国产免费久久久久久婷婷| 亚洲天堂男人天堂女人天堂| 成人高清视频观看www| 韩国福利视频一区| 91精品国产乱码久久久久久蜜臀| 国模私拍一区二区三区| 欧美激情精品久久久| 91九色在线视频| 91在线网站视频| 中文字幕9999| 在线亚洲国产精品网| 国产欧美精品va在线观看| 久久精品国产一区二区三区| 久久久久久一区二区三区| 日韩精品在线免费播放| 精品露脸国产偷人在视频| 久久视频在线直播| 久久噜噜噜精品国产亚洲综合| 色黄久久久久久| 国产精品午夜一区二区欲梦| 在线观看久久av| 久久免费精品日本久久中文字幕| 欧美亚洲免费电影| 日韩国产欧美区| 91精品国产免费久久久久久| 日韩欧美国产成人| 欧洲中文字幕国产精品| 国产日韩在线免费| 日韩成人av在线播放| 国产啪精品视频网站| 97视频免费在线看| 大桥未久av一区二区三区| 成人欧美一区二区三区在线| 国产精品久久久久久网站| 国产精品高潮粉嫩av| 国产精国产精品| 日韩av中文字幕在线免费观看| 中文字幕精品一区久久久久| 日韩一区视频在线| 色综合久久久久久中文网| 欧美日韩激情视频| 久久久成人精品| 日韩免费观看网站| 欧美亚洲成人免费| 国产欧美欧洲在线观看| 亚洲综合色av| 日本精品久久久久久久| 日韩精品极品毛片系列视频| 欧美一区二粉嫩精品国产一线天| 色妞久久福利网| 亚洲最大成人网色| 国产精品久久电影观看| 黄网站色欧美视频| 国内成人精品视频| 国产欧美一区二区三区在线| 久久精品国产96久久久香蕉| 亚洲天堂男人的天堂| 国产免费一区视频观看免费| 伊人精品在线观看| 欧美日本高清一区| 久久免费视频这里只有精品| 中文字幕精品—区二区| 日韩不卡中文字幕| 成人免费视频网| 在线看片第一页欧美| 26uuu国产精品视频| 欧美成人午夜激情视频| 激情亚洲一区二区三区四区| 欧美在线视频观看免费网站| 国产精品视频公开费视频| 亚洲激情在线观看| 最近中文字幕mv在线一区二区三区四区| 亚洲精品在线看| 亚洲国产欧美一区二区丝袜黑人| 国产视频在线一区二区| 欧美人在线视频| 97精品视频在线| 国产成人精品在线|