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

首頁 > 數據庫 > Redis > 正文

分布式鎖定的Redis實現綜述

2020-02-17 14:54:09
字體:
來源:轉載
供稿:網友

在這篇文章里主要想個各位介紹分布式鎖定的Redis實現綜述的相關資料,Redis實現與Zookeeper實現和數據庫實現,供大家交流學習,下面就跟隨武林技術頻道的編輯一起來看看。

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

我們來假設一個最簡單的秒殺場景:數據庫里有一張表,column分別是商品ID,和商品ID對應的庫存量,秒殺成功就將此商品庫存量-1?,F在假設有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*              http://www.49028c.com/article/103617.htm*              http://www.hollischuang.com/archives/1716?utm_source=tuicool&utm_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 RedisCallback() {         @Override         public Boolean doInRedis(RedisConnection redisConnection) throws DataAccessException {           //定義序列化方式           RedisSerializer 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 RedisCallback() {       @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;i<1000;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加鎖成功 ++++ 111111pool-2-thread-5任務執行中9942017-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加鎖成功 ++++ 111111pool-2-thread-6任務執行中9952017-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加鎖成功 ++++ 111111pool-2-thread-19任務執行中9962017-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加鎖成功 ++++ 111111pool-2-thread-11任務執行中9972017-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加鎖成功 ++++ 111111pool-2-thread-4任務執行中9982017-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加鎖成功 ++++ 111111pool-2-thread-1任務執行中9992017-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實現綜述,感謝閱讀,希望能幫助到大家,謝謝大家對武林的支持!
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩午夜在线视频| 日韩av最新在线观看| 国产成人午夜视频网址| 午夜精品一区二区三区视频免费看| 欧美日韩国产一中文字不卡| 日韩av免费网站| 日韩电影免费观看在线| 日本19禁啪啪免费观看www| 国产精品三级网站| 日本精品一区二区三区在线播放视频| 亚洲免费一在线| 亚洲国产精品成人一区二区| 久久精品中文字幕一区| 久久久av免费| 久久免费国产视频| 亚洲自拍小视频免费观看| 国产精品高潮呻吟久久av野狼| 欧美日韩精品在线视频| 国产精品视频男人的天堂| 亚洲国产欧美久久| 亚洲视频在线观看网站| 久久久久国色av免费观看性色| 国产精品久久久久久一区二区| 欧美精品www在线观看| 亚洲精品国产精品国自产观看浪潮| 亚洲片国产一区一级在线观看| 91夜夜未满十八勿入爽爽影院| 亚洲午夜激情免费视频| 国产精品入口尤物| 深夜福利91大全| 久久999免费视频| 91中文字幕在线| 最近2019年中文视频免费在线观看| 中文字幕久热精品在线视频| www.久久久久久.com| 亚洲性夜色噜噜噜7777| 8050国产精品久久久久久| 国产一区二区三区毛片| 欧美美最猛性xxxxxx| 成人网在线观看| 亚洲综合日韩中文字幕v在线| 久久伊人色综合| 亚洲国产中文字幕在线观看| 国产日韩欧美夫妻视频在线观看| 欧美中文在线视频| 国产91精品久| 日韩av一区二区在线| 国产成人涩涩涩视频在线观看| 97涩涩爰在线观看亚洲| 成人国产精品久久久| 日本中文字幕不卡免费| 俺去了亚洲欧美日韩| 午夜精品在线视频| 91牛牛免费视频| 亚洲精选在线观看| 国产精品久久99久久| 成人免费视频97| 综合久久五月天| 欧美国产欧美亚洲国产日韩mv天天看完整| 色综合久综合久久综合久鬼88| 欧美一区二区三区免费观看| 国产精品一区二区久久精品| 亚洲综合精品一区二区| 日韩精品中文字幕久久臀| 国产经典一区二区| 91精品国产综合久久男男| 久久这里有精品视频| 国产精品视频精品视频| 久久综合电影一区| 国产日韩欧美在线观看| 成人天堂噜噜噜| 亚洲 日韩 国产第一| 欧美性生活大片免费观看网址| 久久亚洲欧美日韩精品专区| 91精品国产高清久久久久久久久| 懂色aⅴ精品一区二区三区蜜月| 日韩不卡中文字幕| 夜夜躁日日躁狠狠久久88av| 国产精品一区二区久久国产| 国产亚洲精品久久久久久| 欧美亚洲成人精品| 午夜精品久久17c| 国产成人精品久久亚洲高清不卡| 国产日产欧美a一级在线| 国产亚洲在线播放| 欧美精品一区二区三区国产精品| 亚洲精品视频二区| 国产精品爽爽爽爽爽爽在线观看| 美日韩精品视频免费看| 亚洲第一在线视频| 欧美猛男性生活免费| 北条麻妃99精品青青久久| 亚洲日本aⅴ片在线观看香蕉| 久久精品99国产精品酒店日本| 午夜伦理精品一区| 国产成人精品一区二区| 国产精品海角社区在线观看| 国产日产久久高清欧美一区| 精品国产91久久久| 久久色免费在线视频| 国产精品久久久久久久app| 久久频这里精品99香蕉| 国产美女精彩久久| 成人激情视频小说免费下载| 欧美巨乳美女视频| 欧美激情综合色综合啪啪五月| 亚洲精品久久久久久久久久久久| 2019精品视频| 日韩精品电影网| 国产精品日本精品| 国产精品7m视频| 亚洲乱亚洲乱妇无码| 国产成人亚洲综合青青| 青草青草久热精品视频在线观看| 久久精品国产精品亚洲| 亚洲中国色老太| 亚洲男人天堂久| 欧美精品久久久久久久免费观看| 国产丝袜视频一区| 亚洲精品国产精品国产自| 日韩在线免费观看视频| 在线观看欧美日韩| 日日噜噜噜夜夜爽亚洲精品| 久久在精品线影院精品国产| 国产日韩换脸av一区在线观看| 日韩成人性视频| 人人爽久久涩噜噜噜网站| 国产91精品久久久久久| 成人激情视频小说免费下载| 久久久免费高清电视剧观看| 国产成人精品优优av| 久久久精品影院| 琪琪亚洲精品午夜在线| 美女av一区二区三区| 欧美精品电影在线| 日韩中文av在线| 尤物yw午夜国产精品视频| 国产人妖伪娘一区91| 日本aⅴ大伊香蕉精品视频| 国产精品大片wwwwww| 亚洲性视频网站| 成人网在线视频| 亚洲三级黄色在线观看| 日韩最新在线视频| 国产精品美女免费视频| 久久精品男人天堂| 欧美肥臀大乳一区二区免费视频| 亚洲激情小视频| 亚洲欧美精品一区| 成人福利在线观看| 欧美主播福利视频| 日韩成人激情在线| 亚洲国产精彩中文乱码av在线播放| 亚洲欧美综合v| 欧美激情啊啊啊| 日韩中文字幕网址| 亚洲色图18p| 欧美巨乳在线观看| 国产亚洲欧洲在线| 成人久久久久爱| 国产亚洲福利一区| 国产精品入口日韩视频大尺度| 亚洲成在人线av| 国产成人自拍视频在线观看|