连续登陆活动,或许大家都不会陌生,简单理解就是用户连续登陆了多少天之后,系统就会送一些礼品给相应的用户。最常见的

莫过于游戏和商城这些。游戏就送游戏币之类的东西,商城就送一些礼券。正值国庆,应该也有不少类似的活动。

  下面就对这个的实现提供两个思路,并提供解决方案。

  思路1(以用户为维度):

  连续登陆活动,必然是要求连续登陆,不能有间隔。用1表示登陆,0表示没有登陆,这样我们可以为每个用户创建一个key去存储

他的登陆情况,就可以得到类似这样的一个二进制序列:1110111,如果是7个1,就表示连续7天,如果不是7个1就表示没有连续登

陆7天。所以就能实现这个登陆活动的要求了。

  思路2(以天数为维度):

  一天之内,用户要么是登陆过,要么是没有登陆过。同样的用1来表示登陆过,用0表示没有登陆过。假设我们连续登陆的活动是2天,

同时有3个用户,那么就要有2个key去存储这3个用户的登陆信息,这样就会得到类似这样的两个二进制序列:101(key1),111(key2)。

此时,对这两个key的每一位都进行逻辑与运算,就会得到101,就表明,用户1和用户3连续登陆了两天。从而达到活动的要求。

  之前在string的基础教程中曾经说过关于二进制的相关操作会用一个简单的案例来给大家讲解,现在是兑现这个诺言的时候了。

  下面就简单模拟一下国庆7天假期连续登陆七天的活动。

  方案1 :以用户为维度

  先为每个用户创建一个key(holiday:用户标识),对于我们的例子来说,每个key就会有7位二进制位。这时key会有这样的结构
  

  这时我们就会得到每个用户对应的二进制序列,然后就可以用bitcount命令去得到key含有的1的个数。如果等于7,就是连续登陆了

七天。这样就可以在第七天用户登陆的时间去处理了是否发送礼品了。处理的逻辑是十分简单的。控制器简单逻辑如下:

          [HttpPost]
public IActionResult LoginForEveryone()
{
Random rd = new Random();
var tran = _redis.GetTransaction();
for (int i = ; i < ; i++)
{
for (int j = ; j < ; j++)
{
string activity_key = string.Format("holiday:{0}", j.ToString());
// login 1(true) other 0(false)
if (rd.Next(,) > )
{
tran.StringSetBitAsync(activity_key, i, true);
}
}
}
tran.ExecuteAsync(); List<int> res = new List<int>();
for (int i = ; i < ; i++)
{
string activity_key = string.Format("holiday:{0}", i.ToString());
//7 days
if (_redis.BitCount(activity_key) == )
{
res.Add(i);
}
}
return Json(new { code = "", data = res, count = res.Count });
}
  在这里还是用随机数的方法来模拟数据。主要操作有两个,一个是模拟登陆后把当天对应的偏移设置为1(true),另一个是取出用户

登陆的天数。这是一次性模拟操作,与正常情况的登陆操作还是有些许不同的。大致如下:

         [HttpPost]
public IActionResult LoginForEveryone()
{
//1.login and get the identify of user
//2.get the Current day and write to redis
string activity_key = string.Format("holiday:{0}", "identify of user");
_redis.SetBit(activity_key, currend day, true);
//3.send gift
if(currend day==&& _redis.BitCount(activity_key)==)
{
send gift
}
return ...;
}

  回到我们模拟的情况,在界面展示时,模拟登陆后会显示累计登陆用户的id。

  <script id="everyoneTpl" type="text/html">
<span>total:{{count}}</span>
<ul>
{{each data as item}}
<li>
{{item}}
</li>
{{/each}}
</ul>
</script>
<script>
$(function () {
$("#btn_everyone").click(function () {
$.ajax({
url: "/Holiday/LoginForEveryone",
dataType: "json",
method:"post",
success: function (res) {
if (res.code == "000") {
var html = template('everyoneTpl', res);
$("#div_everyone").html(html);
}
}
})
});
})
</script>
  
  下面来看看效果:
 
  

  演示中:38、103、234、264、412、529这6位用户将得到连续登陆7天的礼品。
 
 
  方案2 :以天数为维度 
  既然是以天数为维度,那么就要定义7个redis的key用来当作每天的登陆记录,类似:
  

  这样的话就要让我们的用户标识是数字才行,如果是用guid做的用户标识就要做一定的处理将其转化成数字,这样方便我们

在给用户设置是否登陆。现在假设我们的用户标识是从1~1000。用户标识对应的就是在key中的偏移量。这时我们就会得到每天

对应的二进制序列,然后就可以用bitop命令去得到逻辑与运算之后的key/value。如果这个key对应偏移量(用户标识)是1,就是

连续登陆了七天,处理的逻辑是十分简单的。控制器简单逻辑如下:

         [HttpPost]
public IActionResult LoginForEveryday()
{
var tran = _redis.GetTransaction(); for (int i = ; i < ; i++)
{
for (int j = ; j < ; j++)
{
//i day,j userId
SetBit(i, j, tran);
}
}
tran.Execute();
//get the result
_redis.BitOP(_bitWise, _res, _redisKeys.ToArray());
IList<int> res = new List<int>();
for (int i = ; i < ; i++)
{
if (_redis.GetBit(_res, i) == true)
{
res.Add(i);
}
}
return Json(new { code = "", data = res, count = res.Count });
} private void SetBit(int day, int userId, StackExchange.Redis.ITransaction tran)
{
switch (day)
{
case :
if (_rd.Next(, ) > )
{
tran.StringSetBitAsync(_first, userId, true);
}
else
{
tran.StringSetBitAsync(_first, userId, false);
}
break;
case :
if (_rd.Next(, ) > )
{
tran.StringSetBitAsync(_second, userId, true);
}
else
{
tran.StringSetBitAsync(_second, userId, false);
}
break;
case :
if (_rd.Next(, ) > )
{
tran.StringSetBitAsync(_thrid, userId, true);
}
else
{
tran.StringSetBitAsync(_thrid, userId, false);
}
break;
case :
if (_rd.Next(, ) > )
{
tran.StringSetBitAsync(_fourth, userId, true);
}
else
{
tran.StringSetBitAsync(_fourth, userId, false);
}
break;
case :
if (_rd.Next(, ) > )
{
tran.StringSetBitAsync(_fifth, userId, true);
}
else
{
tran.StringSetBitAsync(_fifth, userId, false);
}
break;
case :
if (_rd.Next(, ) > )
{
tran.StringSetBitAsync(_sixth, userId, true);
}
else
{
tran.StringSetBitAsync(_sixth, userId, false);
}
break;
case :
if (_rd.Next(, ) >)
{
tran.StringSetBitAsync(_seventh, userId, true);
}
else
{
tran.StringSetBitAsync(_seventh, userId, false);
}
break;
default:
break;
}
}
  
  前台的处理与方案一的一模一样,所以就不贴代码了。下面来看看效果图。
 
  

   可能光看效果图没太大意义,还是要看一下redis中的数据来验证一下的。图中取了76和991这两个用户标识(偏移量)来验证。
  

  可以看到76和991这两个偏移量(用户标识)对应的二进制位是1,也验证了其连续登陆了7天。当然,更多的明细数据也贴出来了

一大堆16进制的东西,有兴趣可以去转换成二进制试试。
 
  对这两种方案简单的总结一下:
 
方案 优点 缺点
以用户为维度 1.可以无缝对接,无论用户标识是数字还是其他
2.key对应的数据较少便于观察
随着用户数量的增长,要管理的key会越来越多
以天数为维度 1.有确定数量的key,方便管理
2.key对应的基数大
1.偏移量可能需要根据实际情况处理
2.数据查看不是很清晰
 

  可至于实际中用那种方案更合适,要根据情况来选择。用户量少的时候,可以用第一种方案,也可以用第二种方案,当用户量很大的时候,

建议采用第二种方案,毕竟只要用户数量没有超过43亿,就不会超出其二进制位数的限制。是可以比较放心使用的。

Redis简单案例(三) 连续登陆活动的简单实现的更多相关文章

  1. Python三次登陆

    题目:Python实现三次登陆 不要急于马上把三次登陆写出来,一定要将复杂的程序简单化,必须一步一步地去扩展,这样才保证不会出错. 步骤一:实现简单的一次登陆 # 事先定义 user = 'dark_ ...

  2. 大数据学习day25------spark08-----1. 读取数据库的形式创建DataFrame 2. Parquet格式的数据源 3. Orc格式的数据源 4.spark_sql整合hive 5.在IDEA中编写spark程序(用来操作hive) 6. SQL风格和DSL风格以及RDD的形式计算连续登陆三天的用户

    1. 读取数据库的形式创建DataFrame DataFrameFromJDBC object DataFrameFromJDBC { def main(args: Array[String]): U ...

  3. SpringDataRedis操作Redis简单案例

    Jedis Jedis是Redis官方推出的一款面向Java的客户端,提供了很多接口供Java语言调用.可以在Redis官网下载,当然还有一些开源爱好者提供的客户端,如Jredis.SRP等等,推荐使 ...

  4. 用redis统计大量用户的登陆情况[只判断是否活跃]

    有这样的一个场景需求:有上亿的用户,要统计这批用户的登陆情况,例如一周连续登陆,连续三天是是否登陆,一周活跃天数等用户 存在的挑战 数据如何尽可能用小的空间存储 如何能快速获取指定的数据 如果使用文件 ...

  5. 案例学python——案例三:豆瓣电影信息入库

    闲扯皮 昨晚给高中的妹妹微信讲题,函数题,小姑娘都十二点了还迷迷糊糊.今天凌晨三点多,被连续的警报声给惊醒了,以为上海拉了防空警报,难不成地震,空袭?难道是楼下那个车主车子被堵了,长按喇叭?开窗看看, ...

  6. Java基础之UDP协议和TCP协议简介及简单案例的实现

    写在前面的废话:马上要找工作了,做了一年的.net ,到要找工作了发现没几个大公司招聘.net工程师,真是坑爹呀.哎,java就java吧,咱从头开始学呗,啥也不说了,玩命撸吧,我真可怜啊. 摘要: ...

  7. zabbix生产环境案例(三)

    生产环境案例(三) 链接:https://pan.baidu.com/s/1q5YwJMTcZLcS5OQ0iOu44A 提取码:8gdi 复制这段内容后打开百度网盘手机App,操作更方便哦 1. Z ...

  8. 搭建一个简单struts2框架的登陆

    第一步:下载struts2对应的jar包,可以到struts官网下载:http://struts.apache.org/download.cgi#struts252 出于学习的目的,可以把整个完整的压 ...

  9. asp:第三平台登陆

    第三平台登陆接口申请网址: http://open.51094.com/ 文档: 第三方合作登录平台使用说明 为方便更多的开发朋友,本人特将当前市面上所有支持第三方联合登录的接口集为一体,以前需要多次 ...

随机推荐

  1. 【原创】免费申请SSL证书【用于HTTPS,即是把网站从HTTP改为HTTPS,加密传输数据,保护敏感数据】

    今天公司有个网站需要改用https访问,所以就用到SSL证书.由于沃通(以前我是在这里申请的)暂停了免费的SSL证书之后,其网站推荐了新的一个网站来申请证书,所以,今天因为刚好又要申请一个证书,所以, ...

  2. springmvc的拦截器

    什么是拦截器                                                         java里的拦截器是动态拦截action调用的对象.它提供了一种机制可以使 ...

  3. SqlServer之数据库三大范式

    分析: 数据库设计应遵循三大范式分别为: 第一范式:确保表中每列的原子性(不可拆分): 第二范式:确保表中每列与主键相关,而不能只与主键的某部分相关(主要针对联合主键),主键列与非主键列遵循完全函数依 ...

  4. MyBatis3.2从入门到精通第一章

    第一章一.引言mybatis是一个持久层框架,是apache下的顶级项目.mybatis托管到goolecode下,再后来托管到github下.(百度百科有解释)二.概述mybatis让程序将主要精力 ...

  5. BZOJ 1597: [Usaco2008 Mar]土地购买 [斜率优化DP]

    1597: [Usaco2008 Mar]土地购买 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4026  Solved: 1473[Submit] ...

  6. 把int*传值给char*,打印出错误的数字

    首先进入debug模式查看i的地址也就是ptr的值 以16进制位小端模式存储(一个整型四个字节,8位16进制数)(根据系统位数情况) 紧接着因为ptr是char*型指针变量,读取数据时按照一个字节一个 ...

  7. [学习笔记]JavaScript之函数式编程

    欢迎指导与讨论:) 前言 函数式编程能使我们的代码结构变得简洁,让代码更接近于自然语言,易于理解. 一.减少不必要的函数嵌套代码 (1)当存在函数嵌套时,若内层函数的参数与外层函数的参数一致时,可以这 ...

  8. LeetCode All in One 题目讲解汇总(持续更新中...)

    终于将LeetCode的免费题刷完了,真是漫长的第一遍啊,估计很多题都忘的差不多了,这次开个题目汇总贴,并附上每道题目的解题连接,方便之后查阅吧~ 477 Total Hamming Distance ...

  9. MySql LIKE 查找带反斜线“\”的记录

    解决方法是在反斜线“\前加“\\\”三个反斜杠. SELECT * FROM 表名 AS a WHERE a.字段 \\\\qc0npwqe.3v4', '%') 原理: 写成三个'\'的原因是反斜线 ...

  10. 龙之谷手游WebVR技术分享

    主要面向Web前端工程师,需要一定Javascript及three.js基础:本文主要分享内容为基于three.js开发WebVR思路及碰到的问题:有兴趣的同学,欢迎跟帖讨论. 目录:一.项目体验1. ...