簽到功能,用 MySQL 還是 Redis ?
關(guān)注我們,設(shè)為星標(biāo),每天7:30不見不散,架構(gòu)路上與您共享 回復(fù)"架構(gòu)師"獲取資源
今天看下簽到功能怎么選擇?
現(xiàn)在的網(wǎng)站和app開發(fā)中,簽到是一個很常見的功能,如微博簽到送積分,簽到排行榜
微博簽到
如移動app ,簽到送流量等活動,
移動app簽到
用戶簽到是提高用戶粘性的有效手段,用的好能事半功倍!
下面我們從技術(shù)方面看看常用的實(shí)現(xiàn)手段:
一. 方案1
直接存到數(shù)據(jù)庫MySQL
用戶表如下:
last_checkin_time 上次簽到時間
checkin_count 連續(xù)簽到次數(shù)
記錄每個用戶簽到信息
簽到流程
1.用戶第一次簽到
last_checkin_time = time()
checkin_count=1
2.用戶非第一次簽到,且當(dāng)天已簽到
什么也不做,返回已簽到。
3.用戶非第一次簽到,且當(dāng)天還未簽到
a.昨天也有簽到
last_checkin_time = time()
checkin_count= checkin_count+1
b.昨天沒有簽到
last_checkin_time = time()
checkin_count=1
使用yii實(shí)現(xiàn)的代碼如下:
//0點(diǎn)
$today_0 = strtotime(date('y-m-d'));
//昨天0點(diǎn)
$yesterday_0 = $today_0-24*60*60;
$last_checkin_time = $model->last_checkin_time;if(empty($last_checkin_time)){
//first checkin
$model->last_checkin_time = time();
$model->checkin_count = 1;
}else{
if($today_0 < $last_checkin_time){
//checkin ed 當(dāng)天已簽到過
return json_encode(['code' => 0, 'msg' => '已簽到成功']);
} //昨天簽到過 if($last_checkin_time < $today_0 && $last_checkin_time > $yesterday_0){
$model->last_checkin_time = time();
$model->checkin_count = $model->checkin_count + 1;
}else{
//昨天沒簽到過,重新計數(shù)
$model->last_checkin_time = time();
$model->checkin_count = 1;
}}$rs = $model->save();
二. 方案2
redis實(shí)現(xiàn)方案,使用bitmap來實(shí)現(xiàn),bitmap是redis 2.2版本開始支持的功能,一般用于標(biāo)識狀態(tài),
另外 ,用bitmap進(jìn)行當(dāng)天有多少人簽到非常的方便,使用bitcount
redis->BITCOUNT($key);
簽到流程
設(shè)置兩個bitmap ,
一個以每天日期為key ,每個uid為偏移量
一個以用戶uid為key ,當(dāng)天在一年中的索引為偏移量,
這樣記錄一個用戶一年的簽到情況僅需要365*1bit
以下是簽到代碼
//每天一個key
$key = 'checkin_' . date('ymd');
if($redis->getbit($key, $uid)){
//已簽到 return json_encode(['code' => 0, 'msg' => '已簽到成功']);
}else{
//簽到 $redis->setbit($key, $uid, 1);
$redis->setbit('checkin_'.$uid , date('z'), 1);
}
以下是用戶連續(xù)簽到計算
public static function getUserCheckinCount($uid){
$key = 'checkin_'.$uid;
$index = date('z');
$n = 0;
for($i = $index; $i>=0;$i--){
$bit = Yii::$app->redis->getbit($key, $i); if($bit == 0) break;
$n++; } return $n;
}
以下是計算一天簽到用戶數(shù)
$key = 'checkin_' . date('ymd');
$redis = Yii::$app->redis;$count = $redis->BITCOUNT($key);
還有什么需求呢?可以自己試著去實(shí)現(xiàn)
三. 優(yōu)缺點(diǎn)比較
1.直接MySQL
思路簡單,容易實(shí)現(xiàn);
缺點(diǎn):占用空間大,表更新比較多,影響性能,數(shù)據(jù)量大時需要用cache輔助;
2.Redis bitmap
優(yōu)點(diǎn)是:
占用空間很小,純內(nèi)存操作,速度快;
缺點(diǎn)是 :
記錄的信息有限,只有一個標(biāo)識位;
偏移量不能大于2^32,512M;大概可以標(biāo)識5億個bit位,絕大多數(shù)的應(yīng)用都是夠用的啦;
偏移量很大的時候可能造成 Redis 服務(wù)器被阻塞;所以要考慮切分。
好啦,兩種方式介紹完了,各有利弊,你喜歡哪種方式呢?
歡迎討論!
文章來源:http://suo.im/5EWN3k
END
順便給大家推薦一個GitHub項(xiàng)目,這個 GitHub 整理了上千本常用技術(shù)PDF,絕大部分核心的技術(shù)書籍都可以在這里找到,
GitHub地址:https://github.com/javadevbooks/books
Gitee地址:https://gitee.com/javadevbooks/books
電子書已經(jīng)更新好了,你們需要的可以自行下載了,記得點(diǎn)一個star,持續(xù)更新中..

