好湿?好紧?好多水好爽自慰,久久久噜久噜久久综合,成人做爰A片免费看黄冈,机机对机机30分钟无遮挡

主頁 > 知識庫 > php基于redis的分布式鎖實例詳解

php基于redis的分布式鎖實例詳解

熱門標簽:百度商鋪地圖標注 常州地圖標注服務商 地圖標注平臺怎么給錢注冊 衡水外呼系統平臺 注冊400電話申請 福州人工外呼系統哪家強 新河科技智能外呼系統怎么樣 釘釘打卡地圖標注 安裝電銷外呼系統

在使用分布式鎖進行互斥資源訪問時候,我們很多方案是采用redis的實現。
固然,redis的單節點鎖在極端情況也是有問題的,假設你的業務允許偶爾的失效,使用單節點的redis鎖方案就足夠了,簡單而且效率高。

redis鎖失效的情況:

客戶端1從master節點獲取了鎖
master宕機了,存儲鎖的key還沒來得及同步到slave節點上
slave升級為master
客戶端2從新的master上獲取到同一個資源的鎖
于是,客戶端1和客戶端2同事持有了同一個資源的鎖,鎖的安全性被打破。
如果我們不考慮這種極端情況,需要實現一個基于單節點redis鎖的大致流程:

set cache_key random_seed NX PX 30000

上面這個set命令拆解開就是:

setnx cache_key random_seed 
expire cache_key 30

雖然這兩組命令執行的效果一樣,但是第二個是非原子性操作,如果執行了setnx成功,但是expire失敗的話,就會造成這個key一直存在了,無法釋放的情況。

redis的作者也指出,在使用單節點redis鎖的時候,設置一個隨機種子作為key的值是很有必要的,保證了一個客戶端釋放的鎖必須是自己所持有的那個鎖。假設獲取鎖時set的不是一個隨機數,而是一個固定值,

那么可能會出現下面的情況:

客戶端1獲取鎖成功
客戶端1在某個操作上阻塞了很長時間
過期時間到了,鎖自動釋放(但是在客戶端1看來自己還是持有鎖中)
客戶端2獲取到了對應同一個資源的鎖
客戶端1從阻塞中恢復了,釋放掉自己持有的鎖,也就是釋放掉了客戶端2持有的鎖
客戶端2的鎖被客戶端1是否,失去安全性。
釋放鎖的操作,很多人直接用del命令,這會有很大的問題,保證不了這個key是被加鎖人鎖刪。這時候需要用到隨機數了。

釋放鎖的操作有三步:

get 所持有鎖
判斷這個鎖是否自己所持有
刪除持有鎖
所以,這三步要保證原子性。用lua腳本來執行,redis官方已經提供腳本文件。

if redis.call("get",KEYS[1]) == ARGV[1] then
  return redis.call("del",KEYS[1])
else
  return 0
end

這段腳本在執行的時候,需要把前面的隨機數作為argv[1] 的值傳進去,把cache_key作為keys[1]的值傳進去。

public class RedisLockHelper {
  @Resource
  private R2mClusterClient r2mClusterClient;

  /**
   * 類似于setNx的功能,同時設置過期時間為expire毫秒
   *
   * @param key  加鎖key
   * @param value 確保在加鎖時間內的唯一因子
   * @param expire 過期時間的毫秒數
   * @return
   */
  private String setLock(String key, String value, long expire) {
    return this.set(key, value, "NX", "PX", expire);
  }

  /**
   * 刪除指定key value
   * 如果 r2m中 key 對應的value==value  返回 1
   * 如果 r2m中 key 對應的value!=value  返回 0
   *
   * @param key
   * @return
   */
  private boolean atomDelete(String key, String value) {
    ListString> values = new ArrayList>();
    values.add(value);
    String sb = "if redis.call('get',KEYS[1])==ARGV[1] then " +
        " return redis.call('del',KEYS[1]) " +
        " else " +
        " return 0" +
        " end";
    if (this.eval(sb, key, values) == 1) {
      return true;
    }
    return false;
  }

  private Long eval(String mobel, String key, ListString> value) {
    return (Long) this.r2mClusterClient.eval(mobel, key, value);
  }

  private String set(String key, String value, String nxxx, String expx, long time) {
    return this.r2mClusterClient.set(key, value, nxxx, expx, time);
  }
}

r2mClusterClient 就是jedis客戶端的封裝。

到此這篇關于php基于redis的分布式鎖實例詳解的文章就介紹到這了,更多相關php基于redis的分布式鎖內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • 詳解PHP解決守護進程Redis假死
  • thinkphp5redis緩存新增方法實例講解
  • PHP使用Redis隊列執行定時任務實例講解
  • Thinkphp5+Redis實現商品秒殺代碼實例講解
  • PHP操作Redis常用命令的實例詳解
  • php在linux環境中如何使用redis詳解
  • php操作redis命令及代碼實例大全
  • php之redis短線重連案例講解

標簽:六安 鶴崗 克拉瑪依 柳州 白城 鷹潭 遼陽 唐山

巨人網絡通訊聲明:本文標題《php基于redis的分布式鎖實例詳解》,本文關鍵詞  php,基于,redis,的,分布式,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《php基于redis的分布式鎖實例詳解》相關的同類信息!
  • 本頁收集關于php基于redis的分布式鎖實例詳解的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 日产无码久久久久久精品通辽团 | 国产又粗又大成人片在线观看| 日本无码免费Av在线观看司葵| 色淫片| 亚洲欧洲精品一区二区三区| 特黄特级特爽视频播放| 男女乱淫真视频免费一级毛片| 黄黄的动漫!| 99热这里都是国产精品| 亚洲欧美在线中文| 深夜在线观看大尺度| 邪恶道r18琉璃神社在线| 超a的校草穿成炮灰omega了!| 快穿肉欲文男主们H| 久久久91精品国产一区二区三区| 亚洲欧美AV无码国产美韩系| gog0西西大胆大尺人体高清| 全色av资源网中文字幕| 欧美女人的逼| 大炕上的肉伦第二部| 国产美女作爱视频| 爽?好大?快?水蜜桃| 老师让女班长脱了内裤打屁股作文 | xxxx色| 日韩精品无码一区二区三区A片 | 太大了坐不下去哈昂| 中文字幕无码亚洲制服在线看小说 | 蜜桃成熟时在线视频| 亚洲人成电影在线看片| 色中射| 91热国内精品永久免费观看| 国产精品亚洲AV三区ai戴眼镜 | 亚洲一区二区三区日本久久九| 日韩美女va毛片在线播放| 人与另类Z0Z??XHD| 内射人妻无码色AB麻豆| 日韩精品中文字幕在线| 娇妻的呻吟~梦颖| 合法共妻(np总受)| 美女张开腿让男人捅| 床戏做爰呻吟声|