Redis實(shí)現(xiàn)CAS的樂(lè)觀鎖

對(duì)于經(jīng)常開(kāi)發(fā)Web的Coder們睬澡,經(jīng)常會(huì)有這樣的需求艇拍,就是在多機(jī)的分布式環(huán)境下,有時(shí)候需要限制多臺(tái)機(jī)器上的請(qǐng)求修改同一份資源骏掀。對(duì)于單機(jī)的環(huán)境下鸠澈,我們通持妫可以用同步或者鎖去避免多線(xiàn)程下的競(jìng)態(tài)條件。以java為例笑陈,我們可以用synchronized或者ReentrantLock际度,去做資源訪(fǎng)問(wèn)的同步。但這是JVM和操作系統(tǒng)提供給我們的特性涵妥,但是對(duì)于分布式環(huán)境下我們沒(méi)有這些便利條件乖菱。所以我們需要引入一個(gè)外部的Observer去實(shí)現(xiàn)這樣的一個(gè)分布式鎖,Zookeeper是一個(gè)比較好的解決方案蓬网,但是Zookeeper還是比較重的窒所,我們可以用Redis實(shí)現(xiàn)這樣一個(gè)鎖。
樂(lè)觀鎖基于CAS思想拳缠,是不具有互斥性墩新,不會(huì)產(chǎn)生鎖等待而消耗資源,但是需要反復(fù)的重試窟坐,但也是因?yàn)橹卦嚨臋C(jī)制海渊,能比較快的響應(yīng)。在實(shí)現(xiàn)CAS之前哲鸳,需要了解一下Redis的事務(wù)機(jī)制臣疑。
Redis事務(wù):
我們可以用Mysql事務(wù)機(jī)制來(lái)理解Redis的事務(wù)機(jī)制,但也有所不同徙菠,Mysql的事務(wù)的形式如下:
openSession()
update()
insert()
commit()
如果在update和insert之間出現(xiàn)錯(cuò)誤讯沈,那么會(huì)觸發(fā)rollback(),Redis的事務(wù)用到了MULTI和EXEC命令婿奔,事務(wù)的形式如下:
MULTI
SET
HSET
EXEC
和Mysql的事務(wù)不同缺狠,Redis會(huì)將所有EXEC命令之前的命令放入一個(gè)QUEUE中,當(dāng)遇到EXEC時(shí)批量執(zhí)行QUEUE中的命令萍摊,但是 Redis的事務(wù)是不支持回滾的挤茄,它只是順序的執(zhí)行命令,并批量返回結(jié)果冰木,但是對(duì)于極端情況下穷劈,事務(wù)在沒(méi)有完全執(zhí)行完時(shí)宕機(jī),導(dǎo)致事務(wù)日志只寫(xiě)入部分踊沸,這樣在重啟時(shí)會(huì)產(chǎn)生錯(cuò)誤歇终,用aof的修復(fù)工具修復(fù)后可以進(jìn)行啟動(dòng)。
在了解了事務(wù)機(jī)制后逼龟,我們還不足以實(shí)現(xiàn)樂(lè)觀鎖评凝,還需要了解一個(gè)命令——Watch,Watch命令可以監(jiān)控Redis中的一個(gè)key腺律,當(dāng)Key發(fā)生變化時(shí)終止事務(wù)的提交肥哎。先看一個(gè)正確的例子:

127.0.0.1:6379> set locktest 1
OK
127.0.0.1:6379> get locktest
"1"
127.0.0.1:6379> watch locktest
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set locktest 3
QUEUED
127.0.0.1:6379> exec
1) OK
127.0.0.1:6379> get locktest
"3"
127.0.0.1:6379> 

但是在multi的過(guò)程中如果locktest的值發(fā)生變化又會(huì)怎樣辽俗?

127.0.0.1:6379> set locktest 1
OK
127.0.0.1:6379> get locktest
"1"
127.0.0.1:6379> watch locktest
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set locktest 3
QUEUED
127.0.0.1:6379> exec
(nil)
127.0.0.1:6379> get locktest
"2"
127.0.0.1:6379> 

這里我們用另一個(gè)Client在Multi之后將locktest修改為2,課件在執(zhí)行事務(wù)的時(shí)候返回為nil篡诽,表示執(zhí)行失敗崖飘。
那么我們就可以用上述兩種命令實(shí)現(xiàn)一個(gè)樂(lè)觀鎖,代碼如下:

package com.redis.lock;

import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.atomic.AtomicInteger;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;
import redis.clients.jedis.Transaction;
/**
 * topic:利用redis的事務(wù)杈女,實(shí)現(xiàn)一個(gè)樂(lè)觀鎖
 * 
 * @author zhiming
 *
 */
public class RedisWatchLock {

    private static final String redisHost = "10.0.5.86";
    
    private static final int port = 6381;
    
    private static JedisPoolConfig config;
    
    private static JedisPool pool;
    
    private static ExecutorService service;
    
    private static int ThLeng=10;
    
    private static CountDownLatch latch;
    
    private static AtomicInteger Countor = new AtomicInteger(0);
    static{
        //利用Redis連接池朱浴,保證多個(gè)線(xiàn)程利用多個(gè)連接,充分模擬并發(fā)性
        config = new JedisPoolConfig();
        config.setMaxIdle(10);
        config.setMaxWaitMillis(1000);
        config.setMaxTotal(30);
        pool = new JedisPool(config, redisHost, port);
        //利用ExecutorService 管理線(xiàn)程
        service = Executors.newFixedThreadPool(10);
        //CountDownLatch保證主線(xiàn)程在全部線(xiàn)程結(jié)束之后退出
        latch = new CountDownLatch(ThLeng);
    }
    
    public static void main(String args[]){
        int ThLeng = 10;
        String ThreadNamePrefix = "thread-";
        Jedis cli = pool.getResource();
        cli.del("redis_inc_key");//先刪除既定的key
        cli.set("redis_inc_key", String.valueOf(1));//設(shè)定默認(rèn)值
        for(int i =0;i<ThLeng;i++){
            Thread th = new Thread(new TestThread(pool));
            th.setName(ThreadNamePrefix+i);
            System.out.println(th.getName()+"inited...");
            service.submit(th);
        }
        service.shutdown();
        try {
            latch.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("all sub thread sucess");
        System.out.println("countor is "+Countor.get());
        String countStr = cli.get("redis_inc_key");
        System.out.println(countStr);
    }
    
    public static class TestThread implements Runnable {
        private String incKeyStr = "redis_inc_key";
        private Jedis cli;
        private JedisPool pool;
        public TestThread(JedisPool pool) {
            cli = pool.getResource();
            this.pool = pool;
            
        }
        public void run() {
            try{
            
                for (int i = 0; i < 100; i++) {
                    actomicAdd();
                }
            }catch(Exception e){
                pool.returnBrokenResource(cli);
            }
            finally{
                pool.returnResource(cli);
                latch.countDown();
            }
        }
        
        public void actomicAdd(){
            boolean flag =true;
            while(flag){
                                cli.watch(incKeyStr);
                String countStr = cli.get("redis_inc_key");
                int countInt = Integer.parseInt(countStr);
                int expect = countInt+1;
                Transaction tx = cli.multi();                   
                tx.set(incKeyStr, String.valueOf(expect));
                List<Object> list = tx.exec();
                //如果事務(wù)失敗了exec會(huì)返回null
                if(list==null){
                    System.out.println("multi shut down");
                    continue;
                }
                else{
                    //如果達(dá)到期望值那么結(jié)束while循環(huán)
                    flag=false;
                }
                System.out.println("my expect num is "+expect);         
                System.out.println("seting....");   
            }
            Countor.incrementAndGet();  
        }
        
    }
    
}

這樣我們就利用Redis實(shí)現(xiàn)了一個(gè)類(lèi)似于Java 的原子類(lèi)的功能达椰。在實(shí)際的Web開(kāi)發(fā)中翰蠢,我們可以利用redis來(lái)解決資源重復(fù)修改或爭(zhēng)用的問(wèn)題。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末啰劲,一起剝皮案震驚了整個(gè)濱河市梁沧,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌蝇裤,老刑警劉巖廷支,帶你破解...
    沈念sama閱讀 211,884評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異栓辜,居然都是意外死亡恋拍,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,347評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén)藕甩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)施敢,“玉大人,你說(shuō)我怎么就攤上這事狭莱〗┩蓿” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 157,435評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵腋妙,是天一觀的道長(zhǎng)悯许。 經(jīng)常有香客問(wèn)我,道長(zhǎng)辉阶,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,509評(píng)論 1 284
  • 正文 為了忘掉前任瘩扼,我火速辦了婚禮谆甜,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘集绰。我一直安慰自己规辱,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,611評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布栽燕。 她就那樣靜靜地躺著罕袋,像睡著了一般改淑。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上浴讯,一...
    開(kāi)封第一講書(shū)人閱讀 49,837評(píng)論 1 290
  • 那天朵夏,我揣著相機(jī)與錄音,去河邊找鬼榆纽。 笑死仰猖,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的奈籽。 我是一名探鬼主播饥侵,決...
    沈念sama閱讀 38,987評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼衣屏!你這毒婦竟也來(lái)了躏升?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,730評(píng)論 0 267
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤狼忱,失蹤者是張志新(化名)和其女友劉穎膨疏,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體藕赞,經(jīng)...
    沈念sama閱讀 44,194評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡成肘,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,525評(píng)論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了斧蜕。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片双霍。...
    茶點(diǎn)故事閱讀 38,664評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖批销,靈堂內(nèi)的尸體忽然破棺而出洒闸,到底是詐尸還是另有隱情,我是刑警寧澤均芽,帶...
    沈念sama閱讀 34,334評(píng)論 4 330
  • 正文 年R本政府宣布丘逸,位于F島的核電站已卸,受9級(jí)特大地震影響哥纫,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜霞丧,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,944評(píng)論 3 313
  • 文/蒙蒙 一劲妙、第九天 我趴在偏房一處隱蔽的房頂上張望湃鹊。 院中可真熱鬧,春花似錦镣奋、人聲如沸币呵。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,764評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)余赢。三九已至芯义,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間妻柒,已是汗流浹背扛拨。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,997評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蛤奢,地道東北人鬼癣。 一個(gè)月前我還...
    沈念sama閱讀 46,389評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像啤贩,于是被迫代替她去往敵國(guó)和親待秃。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,554評(píng)論 2 349

推薦閱讀更多精彩內(nèi)容