解析redis的incr和hash應用

incr

比如北京車牌采取先搶到后審批資質的流程。車牌池子中有n多號碼,頁面呈現以一頁十條的方式展示,每個號碼后有一個搶的按鈕,且一個人只能搶一個車牌,同樣一個車牌只能被一個人搶到。業務模型參考(http://num.10010.com/numapp/chsenumlist/init?num=186)

 if ($this->redis_db->incr("bj_".$car_no) != 1) {       讓別人先下手了,點別的去~   }else{        //搶到競態條件,如果不復核資質要求退出,并清除incr        if(搶到了但是沒資質等){          釋放對此id的竟態權,別占茅坑          $this->yredis_db->del("bj_".$id);        }else{          其他業務A          抱得號碼歸...          其他業務B      } }

另外,incr對string類型,hash類型,sortedSet類型都可以進行操作

blpop

blpop相對于lpop有一個好處,可以對多個隊列進行優先級操作。
blpop會按照key的排列順序依次彈出,返回值為key的listname及具體元素值,而且可以設置block時間,原則是先阻塞先服務.

        $date = date('Ymd', time());         //左進左出 ,優先分配一般的車牌號碼,然后在分配非常好的連號號碼,設置一個阻塞時間         return $this->redis->blpop(self::$_config['dispatch_normal_list'] . $date, self::$_config['dispatch_better_list'] . $date, self::$_config['redis_block_l_pop_time_out']);

hsetnx

設置hash中一個field為指定value,前提是field不存在。如果存在,返回0。
這樣能保證在一個人只能搶一個車牌,但是搶到車牌執行付款或者其他業務操作過程中,其他人無法對此操作,(即不能將此車牌綁定到其他人身上)。根據具體業務情況,可設置基于car_no的hash field和基于 people 的hash field。

hash_base_people {"zhangsan":"京A888","lisi":"京A999"} hash_base_car_no {"京A888":"zhangsan","京A999":"lisi"}

基于這兩個hash 可以做更多關于業務的操作,比如通過hget等查看具體的綁定關系。

hdel

有了通過hsetnx的綁定模型,當某個人對某個車牌交付了訂金等一系列之后,就代表可以永遠的將其消除,這樣會用到hdel。另外如果在指定時間內沒有做比如交付訂金之類的操作,這個車牌號碼會回爐到原始列表中。

 //刪除以people_id為key的hash  $base_people_id_del = $this->redis->hdel(self::$_config['hash_base_people'], $people_id);   //刪除以car_no為key的hash  $base_car_no_del = $this->redis->hdel(self::$_config['hash_base_car_no'], $clue_id);

lpush

如果有入口將北京可以搶拍的車牌放入到一個list里

$lpush_res = $redisObj->lpush($list_name, $car_no);

其中list_name的值可以根據car_no的具體值來確定,比如有6和8的我就放入到better_car_no列表里,其他的放入到normal_car_no列表里,最后可以用blpop來指定一個先后優先級。

rpoplpush

安全的隊列彈出模式,比如N多人對一個入口按鈕進行操作,如果list結構中有足夠的數據,每個人有且只有一條數據會被領取,領取之后再做其他的業務操作。
但是問題是,如果用lpop之后,原隊列中已被彈出,如果中途客戶端在取得該pop的元素后,且完成處理此元素前,客戶端發生崩潰。這時候此條消息就憑空消失了。如果沒有其他補助措施(比如通過綁定或者記錄此彈出的元素)需要嚴謹要求,可以用rpoplpush可以解決此問題。在客戶端真正處理完此pop的元素之后,通過lrem將此消息安全刪除。

推薦學習:《redis視頻教程

以上就是解析

? 版權聲明
THE END
喜歡就支持一下吧
點贊13 分享