springboot项目Redis统计在线用户

springboot项目Redis统计在线用户

我的项目有个显示用户的遗忘曲线,需要统计在线用户以计算他们的曲线

  • 思考了两种方案,但都是用Redis的bitmap数据结构
  • Bitmap是一种特殊类型的数组,其中每个元素只能存储0或1。在Redis中,Bitmap实际上是字符串,每个字符的每一位都被视为一个独立的位,因此一个字符串可以存储多达8*字符串长度的位。 这段代码中,markUserActive方法接收一个用户ID作为参数。然后,它创建一个键,该键由字符串"active_users:"和当前日期组成,格式为ISO_DATE。这样,每天都会有一个新的键,用于存储当天活跃的用户。 然后,它使用redisUtils.setBit方法将用户ID对应的位设置为1。这里,用户ID被用作位的索引。例如,如果用户ID为10,那么第10位将被设置为1。 这样,我们就可以通过检查特定位的值来确定用户是否活跃。如果位的值为1,那么用户就是活跃的;如果位的值为0,那么用户就是不活跃的。 这种方法的优点是,它可以在非常小的空间内存储大量的信息。此外,由于Redis是内存数据库,因此这种方法的速度非常快。
方案一
  • 使用心跳包来追踪和统计用户活跃状态,客户端每隔30分钟或者一段时间给服务端发送一个心跳,服务端获取到用户id然后存储到Redis
  • 但这样的话客户端要定时任务,且依赖客户端。
方案二
  • 在用户每次请求操作的时候,由于我后端配置了Shiro的拦截器判断每次请求是否token过期,加入标记用户活跃的逻辑,并redis设置过期时间2小时
  • 缺点是资源消耗大,每次请求都要标记
@Override
    public void markUserActive(int userId) {
        String key = "active_users:" + LocalDateTime.now().format(DateTimeFormatter.ISO_LOCAL_DATE_TIME);
        redisUtils.setBit(key, userId, true);
        // 设置2小时的过期时间
        redisUtils.expire(key, 2, TimeUnit.HOURS);
    }

markUserActive(int userId) 方法: 这个方法用于标记一个用户为活跃状态。它接收一个用户ID作为参数。方法首先构造一个键,键的格式是 “active_users:” 加上当前的日期和时间。然后,它在 Redis 数据库中将这个键对应的位(由用户ID指定)设置为 true,表示该用户是活跃的。最后,它设置这个键的过期时间为2小时。这意味着,如果2小时内没有再次标记该用户为活跃,那么这个键就会从 Redis 数据库中删除。

@Override
    public List<Integer> getActiveUserIds() {
        List<Integer> activeUserIds = new ArrayList<>();

        // 当前时间
        LocalDateTime currentTime = LocalDateTime.now();
        for (int i = 0; i < 2; i++) {
            // 遍历过去2小时内的键
            String key = "active_users:" + currentTime.minusHours(i).format(DateTimeFormatter.ISO_LOCAL_DATE_TIME);

            if (redisUtils.hasKey(key)) {
                // 遍历1000个用户
                for (int j = 0; j < 1000; j++) {
                    Boolean isUserActive = redisUtils.getBit(key, j);
                    if (isUserActive != null && isUserActive) {
                        activeUserIds.add(j);
                    }
                }
            }
        }

        return activeUserIds;
    }
  • 但是之后测试的时候发现在查找活跃用户的时候并没有找到,可能是时间戳精度问题,所以修改代码将其key保留到分钟形式
    String key = "active_users:" + LocalDateTime.now().truncatedTo(ChronoUnit.MINUTES).format(DateTimeFormatter.ISO_LOCAL_DATE_TIME);
    

相关推荐

  1. springboot项目Redis统计在线用户

    2024-06-06 08:52:01       10 阅读
  2. Springboot项目中对Redis的使用

    2024-06-06 08:52:01       12 阅读
  3. 分布式Shiro,SpringBoot项目Shiro整合Redis

    2024-06-06 08:52:01       9 阅读
  4. springboot项目集成Redis,使用redis各项功能

    2024-06-06 08:52:01       21 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-06-06 08:52:01       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-06-06 08:52:01       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-06-06 08:52:01       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-06-06 08:52:01       20 阅读

热门阅读

  1. 怎么排查native层的bug

    2024-06-06 08:52:01       8 阅读
  2. 【k8s的三种探针】

    2024-06-06 08:52:01       8 阅读
  3. Scala学习笔记7: 对象

    2024-06-06 08:52:01       7 阅读
  4. 小程序真题合集

    2024-06-06 08:52:01       7 阅读
  5. HW面试应急响应之场景题

    2024-06-06 08:52:01       8 阅读
  6. IDEA 开发中一些好用的插件

    2024-06-06 08:52:01       8 阅读
  7. 微信小程序中实现录音功能及其功效

    2024-06-06 08:52:01       6 阅读
  8. 【AI】DeepStream(09):deepstream-app源码详解

    2024-06-06 08:52:01       8 阅读
  9. 从零开始精通Onvif之设备发现

    2024-06-06 08:52:01       13 阅读
  10. 扩展 Kafka 集群从三台节点到四台节点的过程

    2024-06-06 08:52:01       8 阅读