YLBs CAPTCHA - 签到题_一个小小的签到功能,到底用MySQL还是Redis?

b820f1c3fef4d4daf891d746df2d253d.png

来自:网络

今天,看下签到功能怎么选择?

现在的网站和app开发中,签到是一个很常见的功能,如微博签到送积分,签到排行榜~

2d94aa820d12097890f8d15bc638d434.png

微博签到如移动app ,签到送流量等活动,

69496affadd46b8696fe36396c5df26b.png

移动app签到用户签到是提高用户粘性的有效手段,用的好能事半功倍!

下面我们从技术方面看看常用的实现手段:

一. 方案1

直接存到数据库MySQL

用户表如下:

7caf92a3dc57872281166079196a0da8.png

last_checkin_time 上次签到时间

checkin_count 连续签到次数

记录每个用户签到信息

签到流程

1.用户第一次签到

   last_checkin_time = time()
           
    checkin_count=1

2.用户非第一次签到,且当天已签到

什么也不做,返回已签到。

3.用户非第一次签到,且当天还未签到

a.昨天也有签到

   last_checkin_time = time()
           
    checkin_count= checkin_count+1

b.昨天没有签到

   last_checkin_time = time()
           
    checkin_count=1

使用yii实现的代码如下:

   //0点
           
    $today_0 = strtotime(date('y-m-d'));
           
    //昨天0点
           
    $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 当天已签到过
           
     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{
           
     //昨天没签到过,重新计数
           
     $model->last_checkin_time = time();
           
     $model->checkin_count = 1;
           
     }}$rs = $model->save();

二. 方案2

redis实现方案,使用bitmap来实现,bitmap是redis 2.2版本开始支持的功能,一般用于标识状态,

另外 ,用bitmap进行当天有多少人签到非常的方便,使用bitcountcount=redis->BITCOUNT($key);

签到流程

设置两个bitmap ,

  • 一个以每天日期为key ,每个uid为偏移量
  • 一个以用户uid为key ,当天在一年中的索引为偏移量,

这样记录一个用户一年的签到情况仅需要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);
           
     }
       

以下是用户连续签到计算

   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;
           
     }

以下是计算一天签到用户数

   $key = 'checkin_' . date('ymd');
           
    $redis = Yii::$app->redis;$count = $redis->BITCOUNT($key);

还有什么需求呢?可以自己试着去实现。

三. 优缺点比较

1、直接MySQL

思路简单,容易实现;

缺点:占用空间大,表更新比较多,影响性能,数据量大时需要用cache辅助;

2、Redis bitmap

优点是:占用空间很小,纯内存操作,速度快;

缺点是:记录的信息有限,只有一个标识位;

偏移量不能大于2^32,512M;大概可以标识5亿个bit位,绝大多数的应用都是够用的啦;

偏移量很大的时候可能造成 Redis 服务器被阻塞;所以要考虑切分。

好啦,两种方式介绍完了,各有利弊,你喜欢哪种方式呢?

咱们下篇见
每天进步一点点
慢一点才能更快推荐阅读

良月柒:有哪些适合放在程序员办公桌上的高逼格物件?​zhuanlan.zhihu.com
9c774b762fd4dafe3e6b48291e3877ee.png
1k左右的预算,有没有不错的机械键盘推荐?​www.zhihu.com
3cc2a186ed2716280b689cb30f2c2acd.png
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值