欢迎访问悦橙教程(wld5.com),关注java教程。悦橙教程  java问答|  每日更新
页面导航 : > > 文章正文

springboot项目Redis统计在线用户的实现示例,

来源: javaer 分享于  点击 15062 次 点评:199

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

到此这篇关于springboot项目Redis统计在线用户的实现示例的文章就介绍到这了,更多相关springboot Redis统计在线用户内容请搜索3672js教程以前的文章或继续浏览下面的相关文章希望大家以后多多支持3672js教程!

您可能感兴趣的文章:
  • SpringBoot使用Redis的zset统计在线用户信息
相关栏目:

用户点评