做自己的网站有什么用,做网站为职业生存不下去,seo站群优化技术,wordpress32mspringboot项目Redis统计在线用户 我的项目有个显示用户的遗忘曲线#xff0c;需要统计在线用户以计算他们的曲线 思考了两种方案#xff0c;但都是用Redis的bitmap数据结构Bitmap是一种特殊类型的数组#xff0c;其中每个元素只能存储0或1。在Redis中#xff0c;Bitmap实际…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小时缺点是资源消耗大每次请求都要标记
Overridepublic 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 数据库中删除。
Overridepublic ListInteger getActiveUserIds() {ListInteger 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);