校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > Redis實現分布式鎖的幾種方法總結

Redis實現分布式鎖的幾種方法總結

熱門標簽:最簡單的百度地圖標注 小紅書怎么地圖標注店 竹間科技AI電銷機器人 太原營銷外呼系統 地圖標注如何即時生效 玄武湖地圖標注 西藏教育智能外呼系統價格 地圖標注費用 百度商家地圖標注怎么做

Redis實現分布式鎖的幾種方法總結

分布式鎖是控制分布式系統之間同步訪問共享資源的一種方式。在分布式系統中,常常需要協調他們的動作。如果不同的系統或是同一個系統的不同主機之間共享了一個或一組資源,那么訪問這些資源的時候,往往需要互斥來防止彼此干擾來保證一致性,在這種情況下,便需要使用到分布式鎖。

我們來假設一個最簡單的秒殺場景:數據庫里有一張表,column分別是商品ID,和商品ID對應的庫存量,秒殺成功就將此商品庫存量-1。現在假設有1000個線程來秒殺兩件商品,500個線程秒殺第一個商品,500個線程秒殺第二個商品。我們來根據這個簡單的業務場景來解釋一下分布式鎖。

通常具有秒殺場景的業務系統都比較復雜,承載的業務量非常巨大,并發量也很高。這樣的系統往往采用分布式的架構來均衡負載。那么這1000個并發就會是從不同的地方過來,商品庫存就是共享的資源,也是這1000個并發爭搶的資源,這個時候我們需要將并發互斥管理起來。這就是分布式鎖的應用。

1.實現分布式鎖的幾種方案

    1.Redis實現   (推薦)
    2.Zookeeper實現
    3.數據庫實現

Redis實現分布式鎖
*
* 在集群等多服務器中經常使用到同步處理一下業務,這是普通的事務是滿足不了業務需求,需要分布式鎖
*
* 分布式鎖的常用3種實現:
*        0.數據庫樂觀鎖實現
*        1.Redis實現  --- 使用redis的setnx()、get()、getset()方法,用于分布式鎖,解決死鎖問題
*        2.Zookeeper實現
*           參考:http://surlymo.iteye.com/blog/2082684
*              https://www.jb51.net/article/103617.htm
*              http://www.hollischuang.com/archives/1716?utm_source=tuicoolutm_medium=referral
*          1、實現原理:
基于zookeeper瞬時有序節點實現的分布式鎖,其主要邏輯如下(該圖來自于IBM網站)。大致思想即為:每個客戶端對某個功能加鎖時,在zookeeper上的與該功能對應的指定節點的目錄下,生成一個唯一的瞬時有序節點。判斷是否獲取鎖的方式很簡單,只需要判斷有序節點中序號最小的一個。當釋放鎖的時候,只需將這個瞬時節點刪除即可。同時,其可以避免服務宕機導致的鎖無法釋放,而產生的死鎖問題。
2、優點
鎖安全性高,zk可持久化
3、缺點
性能開銷比較高。因為其需要動態產生、銷毀瞬時節點來實現鎖功能。
4、實現
可以直接采用zookeeper第三方庫curator即可方便地實現分布式鎖
*
* Redis實現分布式鎖的原理:
*  1.通過setnx(lock_timeout)實現,如果設置了鎖返回1, 已經有值沒有設置成功返回0
*  2.死鎖問題:通過實踐來判斷是否過期,如果已經過期,獲取到過期時間get(lockKey),然后getset(lock_timeout)判斷是否和get相同,
*   相同則證明已經加鎖成功,因為可能導致多線程同時執行getset(lock_timeout)方法,這可能導致多線程都只需getset后,對于判斷加鎖成功的線程,
*   再加expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS)過期時間,防止多個線程同時疊加時間,導致鎖時效時間翻倍
*  3.針對集群服務器時間不一致問題,可以調用redis的time()獲取當前時間


2.Redis分分布式鎖的代碼實現

  1.定義鎖接口

package com.jay.service.redis; 
 
/** 
 * Redis分布式鎖接口 
 * Created by hetiewei on 2017/4/7. 
 */ 
public interface RedisDistributionLock { 
  /** 
   * 加鎖成功,返回加鎖時間 
   * @param lockKey 
   * @param threadName 
   * @return 
   */ 
  public long lock(String lockKey, String threadName); 
 
  /** 
   * 解鎖, 需要更新加鎖時間,判斷是否有權限 
   * @param lockKey 
   * @param lockValue 
   * @param threadName 
   */ 
  public void unlock(String lockKey, long lockValue, String threadName); 
 
  /** 
   * 多服務器集群,使用下面的方法,代替System.currentTimeMillis(),獲取redis時間,避免多服務的時間不一致問題!!! 
   * @return 
   */ 
  public long currtTimeForRedis(); 
} 

   2.定義鎖實現

package com.jay.service.redis.impl; 
 
import com.jay.service.redis.RedisDistributionLock; 
import org.slf4j.Logger; 
import org.slf4j.LoggerFactory; 
import org.springframework.dao.DataAccessException; 
import org.springframework.data.redis.connection.RedisConnection; 
import org.springframework.data.redis.core.RedisCallback; 
import org.springframework.data.redis.core.StringRedisTemplate; 
import org.springframework.data.redis.serializer.RedisSerializer; 
 
import java.util.concurrent.TimeUnit; 
 
/** 
 * Created by hetiewei on 2017/4/7. 
 */ 
public class RedisLockImpl implements RedisDistributionLock { 
 
  //加鎖超時時間,單位毫秒, 即:加鎖時間內執行完操作,如果未完成會有并發現象 
  private static final long LOCK_TIMEOUT = 5*1000; 
 
  private static final Logger LOG = LoggerFactory.getLogger(RedisLockImpl.class); 
 
  private StringRedisTemplate redisTemplate; 
 
  public RedisLockImpl(StringRedisTemplate redisTemplate) { 
    this.redisTemplate = redisTemplate; 
  } 
 
  /** 
   * 加鎖 
   * 取到鎖加鎖,取不到鎖一直等待知道獲得鎖 
   * @param lockKey 
   * @param threadName 
   * @return 
   */ 
  @Override 
  public synchronized long lock(String lockKey, String threadName) { 
    LOG.info(threadName+"開始執行加鎖"); 
    while (true){ //循環獲取鎖 
      //鎖時間 
      Long lock_timeout = currtTimeForRedis()+ LOCK_TIMEOUT +1; 
      if (redisTemplate.execute(new RedisCallbackBoolean>() { 
        @Override 
        public Boolean doInRedis(RedisConnection redisConnection) throws DataAccessException { 
          //定義序列化方式 
          RedisSerializerString> serializer = redisTemplate.getStringSerializer(); 
          byte[] value = serializer.serialize(lock_timeout.toString()); 
          boolean flag = redisConnection.setNX(lockKey.getBytes(), value); 
          return flag; 
        } 
      })){ 
        //如果加鎖成功 
        LOG.info(threadName +"加鎖成功 ++++ 111111"); 
        //設置超時時間,釋放內存 
        redisTemplate.expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS); 
        return lock_timeout; 
      }else { 
        //獲取redis里面的時間 
        String result = redisTemplate.opsForValue().get(lockKey); 
        Long currt_lock_timeout_str = result==null?null:Long.parseLong(result); 
        //鎖已經失效 
        if (currt_lock_timeout_str != null  currt_lock_timeout_str  System.currentTimeMillis()){ 
          //判斷是否為空,不為空時,說明已經失效,如果被其他線程設置了值,則第二個條件判斷無法執行 
          //獲取上一個鎖到期時間,并設置現在的鎖到期時間 
          Long old_lock_timeout_Str = Long.valueOf(redisTemplate.opsForValue().getAndSet(lockKey, lock_timeout.toString())); 
          if (old_lock_timeout_Str != null  old_lock_timeout_Str.equals(currt_lock_timeout_str)){ 
            //多線程運行時,多個線程簽好都到了這里,但只有一個線程的設置值和當前值相同,它才有權利獲取鎖 
            LOG.info(threadName + "加鎖成功 ++++ 22222"); 
            //設置超時間,釋放內存 
            redisTemplate.expire(lockKey, LOCK_TIMEOUT, TimeUnit.MILLISECONDS); 
 
            //返回加鎖時間 
            return lock_timeout; 
          } 
        } 
      } 
 
      try { 
        LOG.info(threadName +"等待加鎖, 睡眠100毫秒"); 
//        TimeUnit.MILLISECONDS.sleep(100); 
        TimeUnit.MILLISECONDS.sleep(200); 
      } catch (InterruptedException e) { 
        e.printStackTrace(); 
      } 
    } 
  } 
 
  /** 
   * 解鎖 
   * @param lockKey 
   * @param lockValue 
   * @param threadName 
   */ 
  @Override 
  public synchronized void unlock(String lockKey, long lockValue, String threadName) { 
    LOG.info(threadName + "執行解鎖==========");//正常直接刪除 如果異常關閉判斷加鎖會判斷過期時間 
    //獲取redis中設置的時間 
    String result = redisTemplate.opsForValue().get(lockKey); 
    Long currt_lock_timeout_str = result ==null?null:Long.valueOf(result); 
 
    //如果是加鎖者,則刪除鎖, 如果不是,則等待自動過期,重新競爭加鎖 
    if (currt_lock_timeout_str !=null  currt_lock_timeout_str == lockValue){ 
      redisTemplate.delete(lockKey); 
      LOG.info(threadName + "解鎖成功------------------"); 
    } 
  } 
 
  /** 
   * 多服務器集群,使用下面的方法,代替System.currentTimeMillis(),獲取redis時間,避免多服務的時間不一致問題!!! 
   * @return 
   */ 
  @Override 
  public long currtTimeForRedis(){ 
    return redisTemplate.execute(new RedisCallbackLong>() { 
      @Override 
      public Long doInRedis(RedisConnection redisConnection) throws DataAccessException { 
        return redisConnection.time(); 
      } 
    }); 
  } 
 
} 

  3.分布式鎖驗證     

@RestController 
@RequestMapping("/distribution/redis") 
public class RedisLockController { 
 
  private static final String LOCK_NO = "redis_distribution_lock_no_"; 
 
  private static int i = 0; 
 
  private ExecutorService service; 
 
  @Autowired 
  private StringRedisTemplate redisTemplate; 
 
  /** 
   * 模擬1000個線程同時執行業務,修改資源 
   * 
   * 使用線程池定義了20個線程 
   * 
   */ 
  @GetMapping("lock1") 
  public void testRedisDistributionLock1(){ 
 
    service = Executors.newFixedThreadPool(20); 
 
    for (int i=0;i1000;i++){ 
      service.execute(new Runnable() { 
        @Override 
        public void run() { 
          task(Thread.currentThread().getName()); 
        } 
      }); 
    } 
 
  } 
 
  @GetMapping("/{key}") 
  public String getValue(@PathVariable("key") String key){ 
    Serializable result = redisTemplate.opsForValue().get(key); 
    return result.toString(); 
  } 
 
  private void task(String name) { 
//    System.out.println(name + "任務執行中"+(i++)); 
 
    //創建一個redis分布式鎖 
    RedisLockImpl redisLock = new RedisLockImpl(redisTemplate); 
    //加鎖時間 
    Long lockTime; 
    if ((lockTime = redisLock.lock((LOCK_NO+1)+"", name))!=null){ 
      //開始執行任務 
      System.out.println(name + "任務執行中"+(i++)); 
      //任務執行完畢 關閉鎖 
      redisLock.unlock((LOCK_NO+1)+"", lockTime, name); 
    } 
 
  } 
} 

4.結果驗證:

      在Controller中模擬了1000個線程,通過線程池方式提交,每次20個線程搶占分布式鎖,搶到分布式鎖的執行代碼,沒搶到的等待

     結果如下:

2017-04-07 16:27:17.385 INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-4等待加鎖, 睡眠100毫秒
2017-04-07 16:27:17.385 INFO 8652 --- [pool-2-thread-7] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-7解鎖成功------------------
    2017-04-07 16:27:17.391 INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-5加鎖成功 ++++ 111111
pool-2-thread-5任務執行中994
2017-04-07 16:27:17.391 INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-5執行解鎖==========
    2017-04-07 16:27:17.391 INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-1等待加鎖, 睡眠100毫秒
2017-04-07 16:27:17.391 INFO 8652 --- [pool-2-thread-5] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-5解鎖成功------------------
    2017-04-07 16:27:17.397 INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-6加鎖成功 ++++ 111111
pool-2-thread-6任務執行中995
2017-04-07 16:27:17.398 INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-6執行解鎖==========
    2017-04-07 16:27:17.398 INFO 8652 --- [pool-2-thread-6] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-6解鎖成功------------------
    2017-04-07 16:27:17.400 INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-19加鎖成功 ++++ 111111
pool-2-thread-19任務執行中996
2017-04-07 16:27:17.400 INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-19執行解鎖==========
    2017-04-07 16:27:17.400 INFO 8652 --- [ool-2-thread-19] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-19解鎖成功------------------
    2017-04-07 16:27:17.571 INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-11加鎖成功 ++++ 111111
pool-2-thread-11任務執行中997
2017-04-07 16:27:17.572 INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-11執行解鎖==========
    2017-04-07 16:27:17.572 INFO 8652 --- [ool-2-thread-11] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-11解鎖成功------------------
    2017-04-07 16:27:17.585 INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-4加鎖成功 ++++ 111111
pool-2-thread-4任務執行中998
2017-04-07 16:27:17.586 INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-4執行解鎖==========
    2017-04-07 16:27:17.586 INFO 8652 --- [pool-2-thread-4] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-4解鎖成功------------------
    2017-04-07 16:27:17.591 INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-1加鎖成功 ++++ 111111
pool-2-thread-1任務執行中999
2017-04-07 16:27:17.591 INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-1執行解鎖==========
    2017-04-07 16:27:17.591 INFO 8652 --- [pool-2-thread-1] c.jay.service.redis.impl.RedisLockImpl  : pool-2-thread-1解鎖成功------------------

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

您可能感興趣的文章:
  • redis中使用java腳本實現分布式鎖
  • 基于Redis實現分布式鎖以及任務隊列
  • Redis分布式鎖的實現方式(redis面試題)
  • Redis分布式鎖實現方式及超時問題解決
  • Redis上實現分布式鎖以提高性能的方案研究
  • redis實現分布式的方法總結
  • Redis分布式非公平鎖的使用

標簽:澳門 景德鎮 香港 贛州 唐山 揚州 廣東 林芝

巨人網絡通訊聲明:本文標題《Redis實現分布式鎖的幾種方法總結》,本文關鍵詞  Redis,實現,分布式,鎖,的,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Redis實現分布式鎖的幾種方法總結》相關的同類信息!
  • 本頁收集關于Redis實現分布式鎖的幾種方法總結的相關信息資訊供網民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    欧美一级日韩一级| 一区二区三区日韩在线观看| 亚洲欧美日韩久久| 激情欧美一区二区| 欧美色手机在线观看| 国产精品国产三级国产a| 国产呦萝稀缺另类资源| 欧美吞精做爰啪啪高潮| 中文字幕一区二区三中文字幕| 韩国欧美一区二区| 日韩欧美高清在线| 日韩av不卡一区二区| 91精品福利视频| 综合久久一区二区三区| 国产在线精品免费av| 91精品国产福利| 亚洲gay无套男同| 欧美一a一片一级一片| 亚洲色欲色欲www在线观看| 国产成人免费视| 久久久久久亚洲综合影院红桃| 久久精品国产亚洲a| 日韩三级在线观看| 日本伊人色综合网| 91精品婷婷国产综合久久性色 | 欧美色网一区二区| 日韩理论片一区二区| 成人精品视频网站| 国产精品美女久久久久av爽李琼| 国产一区二区三区观看| 精品国产成人在线影院 | 国产亚洲福利社区一区| 国产乱子伦视频一区二区三区| 精品三级在线观看| 国内久久精品视频| 日本一区二区三区国色天香| 国产成人免费视频| 亚洲天堂免费在线观看视频| 日本高清不卡视频| 亚洲午夜羞羞片| 制服视频三区第一页精品| 免费观看91视频大全| 久久久九九九九| 91视视频在线观看入口直接观看www | 天天综合天天综合色| 7777精品伊人久久久大香线蕉超级流畅 | 久久欧美一区二区| 成人一区在线观看| 亚洲精品第1页| 日韩欧美国产精品一区| 国产成人鲁色资源国产91色综 | 精品福利一区二区三区免费视频| 极品销魂美女一区二区三区| 国产午夜精品一区二区三区四区| 丁香另类激情小说| 亚洲午夜一区二区| 久久蜜臀精品av| 91麻豆123| 美女mm1313爽爽久久久蜜臀| 国产亚洲综合av| 在线一区二区视频| 韩国理伦片一区二区三区在线播放| 国产三级欧美三级日产三级99| 91在线精品秘密一区二区| 亚洲成av人片一区二区| 久久久久久久久久久久久久久99 | 理论片日本一区| 中文字幕欧美一| 91精品蜜臀在线一区尤物| 国产精品一区二区你懂的| 亚洲精品欧美激情| 26uuu亚洲婷婷狠狠天堂| 色哟哟精品一区| 国内精品久久久久影院一蜜桃| 亚洲欧洲日韩av| 精品国产一区二区三区久久影院| av不卡免费电影| 久久国内精品自在自线400部| 《视频一区视频二区| 欧美日本国产视频| 99久久精品免费看| 老司机午夜精品99久久| 亚洲一线二线三线久久久| 久久久亚洲午夜电影| 欧美人牲a欧美精品| 96av麻豆蜜桃一区二区| 国产一区啦啦啦在线观看| 亚洲大片免费看| 国产精品电影一区二区| 久久婷婷久久一区二区三区| 欧美性色综合网| 色婷婷久久综合| 成人精品视频一区二区三区 | 精品一区二区在线视频| 一片黄亚洲嫩模| 亚洲视频一二三| 欧美极品少妇xxxxⅹ高跟鞋 | 中文字幕中文字幕一区| 久久综合九色综合欧美就去吻 | 国产亚洲欧美在线| 欧美成人aa大片| 欧美一区二区三区在线观看| 欧美撒尿777hd撒尿| 91女厕偷拍女厕偷拍高清| 国产成人h网站| 国产伦精一区二区三区| 韩国三级中文字幕hd久久精品| 日本中文字幕一区二区视频 | 91精品国产高清一区二区三区蜜臀 | 91女厕偷拍女厕偷拍高清| 丁香婷婷综合色啪| 国产成人精品亚洲777人妖 | 欧美性大战久久| 在线观看亚洲精品视频| 色欲综合视频天天天| 色成年激情久久综合| 91啪亚洲精品| 精品视频1区2区3区| 欧美日韩在线播放一区| 欧美三级电影网站| 7777精品伊人久久久大香线蕉完整版 | 午夜欧美2019年伦理| 日本在线不卡视频一二三区| 精品一区二区三区在线观看国产| 久久99国产精品免费| 国产乱码精品一区二区三| av中文字幕在线不卡| 91精彩视频在线观看| 91精品国产免费久久综合| 久久综合成人精品亚洲另类欧美| 国产人伦精品一区二区| 中文字幕一区免费在线观看| 亚洲同性gay激情无套| 亚洲成人tv网| 激情伊人五月天久久综合| 东方aⅴ免费观看久久av| 色婷婷综合中文久久一本| 欧美精品aⅴ在线视频| 精品日产卡一卡二卡麻豆| 中文字幕精品一区二区三区精品| 亚洲精品菠萝久久久久久久| 日韩影院精彩在线| 成人av影视在线观看| 欧美午夜宅男影院| 久久五月婷婷丁香社区| 亚洲色图19p| 狠狠色狠狠色综合| 91丝袜呻吟高潮美腿白嫩在线观看| 欧美日韩高清影院| 国产亚洲1区2区3区| 亚洲国产精品综合小说图片区| 久久99精品国产.久久久久 | 伊人性伊人情综合网| 久久精品99国产国产精| 99久久亚洲一区二区三区青草| 91精品国产欧美日韩| 亚洲免费观看高清在线观看| 日韩avvvv在线播放| 91美女在线视频| 久久亚洲影视婷婷| 亚洲.国产.中文慕字在线| 成人午夜又粗又硬又大| 欧美一级日韩不卡播放免费| 亚洲免费观看在线观看| 国产伦精品一区二区三区视频青涩| 欧美自拍丝袜亚洲| 日本美女一区二区三区视频| 成人性视频免费网站| 国产日产精品1区| 日本一区二区三区久久久久久久久不 | 日本aⅴ亚洲精品中文乱码| 成人午夜免费av| 欧美mv和日韩mv的网站| 午夜国产精品一区| 91视频精品在这里| 国产精品女主播av| 国产一区二区三区综合| 日韩免费视频一区| 日韩av电影免费观看高清完整版在线观看 | 欧美美女黄视频| 中文字幕视频一区二区三区久| 激情文学综合丁香| 91精品国产综合久久久久久久| 亚洲黄一区二区三区| av电影一区二区| 亚洲国产高清不卡| 国产黑丝在线一区二区三区| 日韩免费观看高清完整版| 亚洲成人福利片| 欧美色综合天天久久综合精品| 亚洲女子a中天字幕| 99精品欧美一区| 亚洲精选在线视频| 91在线视频在线| 一区二区免费在线播放| 色天使久久综合网天天| 一区二区三区精品在线观看| 色视频一区二区| 午夜精品免费在线| 欧美精品久久99久久在免费线 |