Redis 乐观锁详解及应用

havepop 4年前
   <p>在Redis的事务中,WATCH命令可用于提供CAS(check-and-set)功能。假设通过WATCH命令在事务执行之前监控了某个key,倘若在WATCH之后Key的值发生了变化,EXEC命令执行的事务将被放弃,同时返回nil以通知调用者事务执行失败:</p>    <pre>  <code class="language-html">redis> SET key 1  OK  redis> WATCH key  OK  redis> SET key 2  OK  redis> MULTI  OK  redis> SET key 3  QUEUED  redis> EXEC  (nil)  redis> GET key  "2"</code></pre>    <p>因此,借用redis使用watch可以完成秒杀抢购功能,使用redis中两个key完成秒杀抢购功能,mywatchkey用于存储抢购数量和mywatchlist用户存储抢购列表。</p>    <p>php示例代码:</p>    <pre>  <code class="language-html"><?php    $redis = new redis();    $result = $redis->connect('127.0.0.1', 6379);    $mywatchkey = $redis->get("mywatchkey");    $rob_total = 100;   //抢购数量    if($mywatchkey<$rob_total){        $redis->watch("mywatchkey");        $redis->multi();                //设置延迟,方便测试效果。        sleep(5);        //插入抢购数据        $redis->hSet("mywatchlist","user_id_".mt_rand(1, 9999),time());        $redis->set("mywatchkey",$mywatchkey+1);        $rob_result = $redis->exec();        if($rob_result){            $mywatchlist = $redis->hGetAll("mywatchlist");            echo "抢购成功!<br/>";            echo "剩余数量:".($rob_total-$mywatchkey-1)."<br/>";            echo "用户列表:<pre>";            var_dump($mywatchlist);  //打印抢购成功用户      }else{            echo "手气不好,再抢购!";exit;        }    }</code></pre>    <p> </p>    <p>来自: <a href="/misc/goto?guid=4959674034057335420" rel="nofollow">http://www.36nu.com/post/176.html</a></p>    <p> </p>