一个小小的签到功能,到底用MySQL还是Redis

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
简介: 一个小小的签到功能,到底用MySQL还是Redis

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

 

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

 

微博签到

 

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

 

移动app签到

 

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

 

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

 

一. 方案1

 

直接存到数据库MySQL

 

用户表如下:

 

 

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进行当天有多少人签到非常的方便,使用bitcount

count=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 服务器被阻塞;所以要考虑切分。

 

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

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
相关文章
|
17天前
|
NoSQL Java 关系型数据库
Liunx部署java项目Tomcat、Redis、Mysql教程
本文详细介绍了如何在 Linux 服务器上安装和配置 Tomcat、MySQL 和 Redis,并部署 Java 项目。通过这些步骤,您可以搭建一个高效稳定的 Java 应用运行环境。希望本文能为您在实际操作中提供有价值的参考。
89 26
|
3天前
|
NoSQL 关系型数据库 MySQL
Linux安装jdk、mysql、redis
Linux安装jdk、mysql、redis
59 7
|
1月前
|
缓存 NoSQL 关系型数据库
Redis和Mysql如何保证数据⼀致?
在项目中,为了解决Redis与Mysql的数据一致性问题,我们采用了多种策略:对于低一致性要求的数据,不做特别处理;时效性数据通过设置缓存过期时间来减少不一致风险;高一致性但时效性要求不高的数据,利用MQ异步同步确保最终一致性;而对一致性和时效性都有高要求的数据,则采用分布式事务(如Seata TCC模式)来保障。
68 14
|
1月前
|
存储 NoSQL 关系型数据库
MySQL和Redis的区别
**MySQL和Redis的区别** MySQL和Redis都是流行的数据存储解决方案,但它们在设计、用途和特性上有显著区别。理解这些区别有助于选择合适的数据库来满足不同的应用需求。本文将详细介绍MySQL和Redis的区别,包括它们的架构、使用场景、性能和其他关键特性。 ### 一、基本概述 **MySQL**: MySQL是一个关系型数据库管理系统(RDBMS),使用结构化查询语言(SQL)进行数据管理。它支持事务、复杂查询和多种存储引擎,广泛应用于各种Web应用、企业系统和数据分析项目。 **Redis**: Redis是一个基于内存的键值数据库,通常被称为NoSQL数
95 4
|
1月前
|
存储 NoSQL PHP
如何用Redis高效实现点赞功能?用Set?还是Bitmap?
在众多软件应用中,点赞功能几乎成为标配。本文从实际需求出发,探讨如何利用 Redis 的 `Set` 和 `Bitmap` 数据结构设计高效点赞系统,分析其优缺点,并提供 PHP 实现示例。通过对比两种方案,帮助开发者选择最适合的存储方式。
42 3
|
1月前
|
SQL NoSQL 关系型数据库
2024Mysql And Redis基础与进阶操作系列(13)作者——LJS[你个小黑子这都还学不会嘛?你是真爱粉嘛?真是的 ~;以后请别侮辱我家鸽鸽]
MYSQL日志之详解如何配置查看二进制、查询及慢查询日志;备份与恢复等具体详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
2024Mysql And Redis基础与进阶操作系列(13)作者——LJS[你个小黑子这都还学不会嘛?你是真爱粉嘛?真是的 ~;以后请别侮辱我家鸽鸽]
|
6天前
|
存储 缓存 NoSQL
解决Redis缓存数据类型丢失问题
解决Redis缓存数据类型丢失问题
122 85
|
2月前
|
消息中间件 缓存 NoSQL
Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。
【10月更文挑战第4天】Redis 是一个高性能的键值对存储系统,常用于缓存、消息队列和会话管理等场景。随着数据增长,有时需要将 Redis 数据导出以进行分析、备份或迁移。本文详细介绍几种导出方法:1)使用 Redis 命令与重定向;2)利用 Redis 的 RDB 和 AOF 持久化功能;3)借助第三方工具如 `redis-dump`。每种方法均附有示例代码,帮助你轻松完成数据导出任务。无论数据量大小,总有一款适合你。
84 6
|
3天前
|
缓存 监控 NoSQL
Redis经典问题:缓存穿透
本文详细探讨了分布式系统和缓存应用中的经典问题——缓存穿透。缓存穿透是指用户请求的数据在缓存和数据库中都不存在,导致大量请求直接落到数据库上,可能引发数据库崩溃或性能下降。文章介绍了几种有效的解决方案,包括接口层增加校验、缓存空值、使用布隆过滤器、优化数据库查询以及加强监控报警机制。通过这些方法,可以有效缓解缓存穿透对系统的影响,提升系统的稳定性和性能。
|
1月前
|
缓存 NoSQL 关系型数据库
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
本文详解缓存雪崩、缓存穿透、缓存并发及缓存预热等问题,提供高可用解决方案,帮助你在大厂面试和实际工作中应对这些常见并发场景。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题