您当前的位置:首页 > 电脑百科 > 数据库 > Redis

Redis实现统计网站访问人数的功能

时间:2019-08-19 13:20:06  来源:  作者:
Redis实现统计网站访问人数的功能

 

现在想要统计某一网站的累积访问用户人数和日均活跃人数(连续多少天访问该网站的人数),可以通过redis来实现类似功能。

笔者使用的数据结构是Redis中的bitmap,其在大数据量下的空间占用量很小。大概思路就是每一位用户都是bitmap中的一位,为1就代表其访问了,为0就代表没访问。比如说现在有5位用户,第1、3位用户访问了,而2、4、5没访问,如果以索引位置作为其userId的话,那么bitmap存储的就是10100。

累计用户的key设置为“totalKey”,其值为到今天为止所有用户访问的信息,为1就代表其访问过该网站,为0就代表该用户直到今天都没有访问过该网站;日均活跃人数的key设置为“activeKey:[当前的日期]”,比如说2019年5月31日的日均活跃人数key为“activeKey:20190531”,2019年5月30日的日均活跃人数key为“activeKey:20190530”,等等。所以如果要统计日均活跃人数的话,只要将这几个key做交集就可以了(因为只有都为1,相与后结果才为1,如果有一个为0,相与后结果就不是1),然后统计交集结果的1的个数,结果即为统计值。

实现代码如下所示,在main函数中模拟了用户访问的情况。在2019年5月31日有userId为0到14一共15个人访问该网站,而在2019年5月30日有userId为6到14一共9个人访问过该网站:

package com.htli.redis;
 
import JAVA.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
 
import org.Apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.time.DateUtils;
import org.apache.commons.pool2.impl.GenericObjectPoolConfig;
 
import redis.clients.jedis.BitOP;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.Pipeline;
 
/**
 * 统计累计和日均活跃用户人数
 * @author Robert Hou
 * @date 2019年5月31日
 */
public class Counter {
 
 /**
 * ip地址
 */
 private static final String IP_ADDRESS = "192.168.253.129";
 /**
 * 端口号
 */
 private static final int PORT = 6379;
 /**
 * jedis客户端
 */
 private Jedis jedis;
 /**
 * 累计用户人数key
 */
 private static final String TOTAL_KEY = "totalKey";
 /**
 * 日均活跃用户人数key
 */
 private static final String ACTIVE_KEY = "activeKey:";
 
 public Counter() {
 GenericObjectPoolConfig poolConfig = new GenericObjectPoolConfig();
 poolConfig.setMaxTotal(50);
 poolConfig.setMaxIdle(50);
 poolConfig.setMaxWaitMillis(1000);
 JedisPool jedisPool = new JedisPool(poolConfig, IP_ADDRESS, PORT);
 jedis = jedisPool.getResource();
 }
 
 /**
 * 更新累计和日均活跃用户人数
 * @param userId 用户id
 * @param time 当前日期
 */
 private void updateUser(long userId, String time) {
 if (StringUtils.isBlank(time)) {
 SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd");
 time = sdf.format(new Date());
 }
 Pipeline pipeline = jedis.pipelined();
 pipeline.setbit(TOTAL_KEY, userId, true);
 pipeline.setbit(ACTIVE_KEY + time, userId, true);
 pipeline.syncAndReturnAll();
 }
 
 /**
 * 获取累计用户人数
 * @return 累计用户人数
 */
 private Long getTotalUserCount() {
 Pipeline pipeline = jedis.pipelined();
 pipeline.bitcount(TOTAL_KEY);
 List<Object> totalKeyCountList = pipeline.syncAndReturnAll();
 return (Long) totalKeyCountList.get(0);
 }
 
 /**
 * 获取指定天数内的日均活跃人数
 * @param dayNum 指定天数
 * @return 
 */
 private Long getActiveUserCount(int dayNum) {
 if (dayNum < 1) {
 return (long) 0;
 }
 List<String> pastDaysKey = new ArrayList<>();
 SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd");
 StringBuilder sb = new StringBuilder();
 for (int i = 0; i < dayNum; i++) {
 //保存距今dayNum天数的key的集合
 sb.Append(ACTIVE_KEY).append(sdf.format(DateUtils.addDays(new Date(), -i)));
 pastDaysKey.add(sb.toString());
 sb.delete(0, sb.length());
 }
 if (pastDaysKey.isEmpty()) {
 return (long) 0;
 }
 String lastDaysKey = "last" + dayNum + "DaysActive";
 Pipeline pipeline = jedis.pipelined();
 pipeline.bitop(BitOP.AND, lastDaysKey, pastDaysKey.toArray(new String[pastDaysKey.size()]));
 pipeline.bitcount(lastDaysKey);
 //设置过期时间为5分钟
 pipeline.expire(lastDaysKey, 300);
 List<Object> activeKeyCountList = pipeline.syncAndReturnAll();
 return (Long) activeKeyCountList.get(1);
 }
 
 public static void main(String[] args) {
 Counter c = new Counter();
 for (int i = 0; i < 15; i++) {
 c.updateUser(i, "20190531");
 }
 for (int i = 6; i < 15; i++) {
 c.updateUser(i, "20190530");
 }
 System.out.println("累计用户数:" + c.getTotalUserCount());
 System.out.println("两天内的活跃人数:" + c.getActiveUserCount(2));
 }
}

运行结果如下所示:

累计用户数:15

两天内的活跃人数:9



Tags:Redis   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
来源: my.oschina.net/xiaomu0082/blog/2990388首先说下问题现象:内网sandbox环境API持续1周出现应用卡死,所有api无响应现象刚开始当测试抱怨环境响应慢的时候 ,我们重启一下应...【详细内容】
2021-12-08  Tags: Redis  点击:(18)  评论:(0)  加入收藏
我不知道为什么你会选择对特定数量的“错误”(或警告)如此具体。听起来您正在寻找将要发布到 Yahoo! 的某些文章的内容。 Insider (N Foos to Blah for the BlahBlah)。那说:...【详细内容】
2021-12-07  Tags: Redis  点击:(14)  评论:(0)  加入收藏
目录 一、背景 二、步骤 0.理论支持 1、获取数据 2、结果 3、分析数据并评估大小 三、关于repl-backlog-size 一、背景 repl-backlog-size控制这个环形缓冲区. ​ 主从断...【详细内容】
2021-11-05  Tags: Redis  点击:(41)  评论:(0)  加入收藏
Redis 性能测试是通过同时执行多个命令实现的。1,Redis-benchmarkRedis性能命令:redis性能命令格式: redis-benchmark [option] [option value] redis 性能测试工具可选参数如...【详细内容】
2021-11-02  Tags: Redis  点击:(41)  评论:(0)  加入收藏
1 概述数据结构和内部编码 无传统关系型数据库的 Table 模型schema 所对应的db仅以编号区分。同一 db 内,key 作为顶层模型,它的值是扁平化的。即 db 就是key的命名空间。 key...【详细内容】
2021-11-01  Tags: Redis  点击:(28)  评论:(0)  加入收藏
普通java中使用引用Java redis 驱动,即可连接:import redis.clients.jedis.Jedis; public class RedisTestJava { public static void main(String[] args) { //连...【详细内容】
2021-10-13  Tags: Redis  点击:(34)  评论:(0)  加入收藏
Redis常用的数据结构有 string list set zset hashstringstring 是 Redis 的基本的数据类型,一个 key 对应一个 value。string 类型是二进制安全的,Redis的string可以包含任...【详细内容】
2021-10-12  Tags: Redis  点击:(36)  评论:(0)  加入收藏
列表类型可以存储一组按插入顺序排序的字符串,它非常灵活,支持在两端插入、弹出数据,可以充当栈和队列的角色。> LPUSH fruit apple(integer) 1> RPUSH fruit banana(integer)...【详细内容】
2021-09-17  Tags: Redis  点击:(54)  评论:(0)  加入收藏
Redis持久化意义 是做灾难恢复,数据恢复,也可以归类到高可用的一个环节里面去,比如你的redis整个挂了,然后redis就不可用了,你要做的事情是让redis变得可用,尽快变得可用 大量的请...【详细内容】
2021-08-12  Tags: Redis  点击:(77)  评论:(0)  加入收藏
Nginx来限制访问控制的方法有多种,nginx主要有2个模块控制,但是那些不支持自定义,非常死,在大多数场景下并不实用。今天分享一个:利用openresty+lua+redis 实现封杀频繁恶意访问I...【详细内容】
2021-08-12  Tags: Redis  点击:(119)  评论:(0)  加入收藏
▌简易百科推荐
来源: my.oschina.net/xiaomu0082/blog/2990388首先说下问题现象:内网sandbox环境API持续1周出现应用卡死,所有api无响应现象刚开始当测试抱怨环境响应慢的时候 ,我们重启一下应...【详细内容】
2021-12-08  Java识堂    Tags:Redis   点击:(18)  评论:(0)  加入收藏
我不知道为什么你会选择对特定数量的“错误”(或警告)如此具体。听起来您正在寻找将要发布到 Yahoo! 的某些文章的内容。 Insider (N Foos to Blah for the BlahBlah)。那说:...【详细内容】
2021-12-07  富集云科技有限公司    Tags:Redis   点击:(14)  评论:(0)  加入收藏
目录 一、背景 二、步骤 0.理论支持 1、获取数据 2、结果 3、分析数据并评估大小 三、关于repl-backlog-size 一、背景 repl-backlog-size控制这个环形缓冲区. ​ 主从断...【详细内容】
2021-11-05  弈秋的美好生活    Tags:redis   点击:(41)  评论:(0)  加入收藏
Redis 性能测试是通过同时执行多个命令实现的。1,Redis-benchmarkRedis性能命令:redis性能命令格式: redis-benchmark [option] [option value] redis 性能测试工具可选参数如...【详细内容】
2021-11-02  川石信息    Tags:Redis   点击:(41)  评论:(0)  加入收藏
1 概述数据结构和内部编码 无传统关系型数据库的 Table 模型schema 所对应的db仅以编号区分。同一 db 内,key 作为顶层模型,它的值是扁平化的。即 db 就是key的命名空间。 key...【详细内容】
2021-11-01  JavaEdge    Tags:Redis   点击:(28)  评论:(0)  加入收藏
普通java中使用引用Java redis 驱动,即可连接:import redis.clients.jedis.Jedis; public class RedisTestJava { public static void main(String[] args) { //连...【详细内容】
2021-10-13  faesuite    Tags:Redis   点击:(34)  评论:(0)  加入收藏
Redis常用的数据结构有 string list set zset hashstringstring 是 Redis 的基本的数据类型,一个 key 对应一个 value。string 类型是二进制安全的,Redis的string可以包含任...【详细内容】
2021-10-12  语霖    Tags:Redis   点击:(36)  评论:(0)  加入收藏
列表类型可以存储一组按插入顺序排序的字符串,它非常灵活,支持在两端插入、弹出数据,可以充当栈和队列的角色。> LPUSH fruit apple(integer) 1> RPUSH fruit banana(integer)...【详细内容】
2021-09-17  深夜敲代码    Tags:Redis   点击:(54)  评论:(0)  加入收藏
Redis持久化意义 是做灾难恢复,数据恢复,也可以归类到高可用的一个环节里面去,比如你的redis整个挂了,然后redis就不可用了,你要做的事情是让redis变得可用,尽快变得可用 大量的请...【详细内容】
2021-08-12  小李说IT    Tags:Redis   点击:(77)  评论:(0)  加入收藏
当查询Redis中没有的数据时,该查询会下沉到数据库层,同时数据库层也没有该数据,当这种情况大量出现或被恶意攻击时,接口的访问全部透过Redis访问数据库,而数据库中也没有这些数据...【详细内容】
2021-07-30  随便t    Tags:缓存穿透   点击:(91)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条