统计在线用户的数量,是应用很常见的需求了。如果需要精准的统计到用户是在线,离线状态,我想只有客户端和服务器通过保持一个TCP长连接来实现。如果应用本身并非一个IM应用的话,这种方式成本极高。

现在的应用都趋向于使用心跳包来标识用户是否在线。用户登录后,每隔一段时间,往服务器推送一个消息,表示当前用户在线。服务器则可以定义一个时间差,例如:5分钟内收到过客户端心跳消息,视为在线用户。

在线用户统计的实现

基于数据库实现

最简单的办法,就是在用户表,添加一个最后心跳包的日期时间字段 last_active。服务器收到心跳后,每次都去更新这个字段为当前的最新时间。

如果要查询最近5分钟活跃的用户数量,就可以简单的通过一句SQL完成。SELECT COUNT(1) AS `online_user_count` FROM `user` WHERE `last_active` BETWEEN  '2020-12-22 13:00:00' AND '020-12-22 13:05:00';

弊端也是显而易见,为了提高检索效率,不得不为last_active字段添加索引,而因为心跳的更新,会导致频繁的重新维护索引树,效率极其低下。

基于Redis实现

这是比较理想的一种实现方式了,Redis基于内存进行读写,性能自然比关系型数据库好得多,而且它所提供的Zset可以很方便的构建出一个在线用户的统计服务。

Redis的Zset

这里不会涉及太多redis的东西,简单说明以下zset。它是一个有序的set集合,集合中的每个元素由2个东西组成member 既然是集合,那么它便是集合中的元素,并且不能重复

score  既然是有序的,它就是用于排序的权重字段

Zset的部分操作

添加元素ZADD key score member [score member ...]

一次性添加一个或者多个元素到集合,如果member已经存在则会使用当前score进行覆盖

统计所有的元素数量ZCARD key

统计score值在min和max之间元素数量ZCOUNT key min max

删除score值在min和max之间的元素ZREMRANGEBYSCORE key min max

一个示例

我打算,用一个zset存储我内心中编程语言的评分排名,这个key叫做lang

添加信息,返回新添加的元素个数> zadd lang 999 php 10 java 9 go 8 python 7 javascript

"5"

查看添加的数量> zcard lang

"5"

查看评分在8 - 10之间的元素个数,有3个> zcount lang 8 10

"3"

删除评分在8 - 1000的元素,返回删除的个数> ZREMRANGEBYSCORE lang 8 1000

"4"

在线用户服务的实现

知道了zset后,就可以实现一个在线用户的统计服务了。

实现思路

客户端每隔5分钟发送一个心跳到服务器,服务器根据会话获取到用户的ID,作为zset的member

存入zset,score便是当前收到心跳的时间戳,当同一个用户第二次发送心跳的时候,就会更新他对应的score值,由于更新是在内存,这个速度相当快。zadd users 1608616915109 10000

需要统计出在线用户的数量,本质上就是需要统计出,最近5分钟有发送心跳的用户,通过zcount可以很轻松的统计出来。通过程序获取到当前的时间戳,作为maxScore,时间戳减去5分钟后作为minScore。zcount users 1608616615109 1608616915109

因为某些用户可能长时间没有登录过了,可以通过ZREMRANGEBYSCORE进行清理。通过程序获取到当前的时间戳,减去5分钟后作为maxScore,使用0, 作为minScore,表示清理所有超过5分钟没有发送过心跳包的用户。ZREMRANGEBYSCORE users 0 1608616615109

实现代码import java.time.Duration;

import java.time.Instant;

import java.time.LocalDateTime;

import java.time.ZoneId;

import javax.annotation.Resource;

import org.springframework.data.redis.core.StringRedisTemplate;

import org.springframework.stereotype.Component;

/**

*

*

* 在线用户统计

*

* @author Administrator

*

*/

@Component

public class OnlineUserStatsService {

private static final String ONLINE_USERS = "onlie_users";

@Resource

private StringRedisTemplate stringRedisTemplate;

/**

* 添加用户在线信息

* @param userId

* @return

*/

public Boolean online(Integer userId) {

return this.stringRedisTemplate.opsForZSet().add(ONLINE_USERS, userId.toString(), Instant.now().toEpochMilli());

}

/**

* 获取一定时间内,在线的用户数量

* @param duration

* @return

*/

public Long count(Duration duration) {

LocalDateTime now = LocalDateTime.now();

return this.stringRedisTemplate.opsForZSet().count(ONLINE_USERS,

now.minus(duration).atZone(ZoneId.systemDefault()).toInstant().toEpochMilli(),

now.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli());

}

/**

* 获取所有在线过的用户数量,不论时间

* @return

*/

public Long count() {

return this.stringRedisTemplate.opsForZSet().zCard(ONLINE_USERS);

}

/**

* 清除超过一定时间没在线的用户数据

* @param duration

* @return

*/

public Long clear(Duration duration) {

return this.stringRedisTemplate.opsForZSet().removeRangeByScore(ONLINE_USERS, 0,

LocalDateTime.now().minus(duration).atZone(ZoneId.systemDefault()).toInstant().toEpochMilli());

}

}

使用示例@Resource

private OnlineUserStatsService onlineUserStatsService;

@Test

public void test() {

// ID为1的用户发送了心跳包

boolean result = this.onlineUserStatsService.online(1);

System.out.println("online=" + result);

// 获取5分钟内,发送过心跳包的用户数量,也就是在线用户的数量

Long count = this.onlineUserStatsService.count(Duration.ofMinutes(5));

System.out.println("oneline count=" + count);

// 获取所有发送过心跳包的用户数量

count = this.onlineUserStatsService.count();

System.out.println("all count=" + count);

// 清除超过1天都没发送过心跳包的用户

Long clear = this.onlineUserStatsService.clear(Duration.ofDays(1));

System.out.println("clear=" + clear);

}

内存消耗分析

我对Redis的内存分配并不熟悉,只是按照自己的想法去填写了一些数据,所以我在这里理解的东西,可能是错误的。但是我想这并不耽误证明 - 在这种场景使用Zset对内存消耗极低的事实

设想onlie_users需要存储1亿个用户的状态信息,每个元素score和member需要10个字节存储,那么一共大约需要20G内存。20G的内存对于现在的服务器来说,并不是大问题。

最后心跳协议不一定非要HTTP,如果客户端支持的话UDP就很适合,可以节约一些系统开销。

zset的key,不一定非要用String,可以修改序列化方式,以固定的字节的形式存储用户ID,在用户ID过大的时候,可以节约一些存储空间。String userId = "10010";

System.out.println(userId.getBytes().length); // 以字符串形式存储 => 需要5个字节

byte[] bin = ByteBuffer.allocate(4).putInt(Integer.valueOf(userId)).array();

System.out.println(bin.length);                    // 序列化为字节形式存储 => 需要4个字节

System.out.println(ByteBuffer.wrap(bin).getInt());    // 反序列化为ID => 10010

java和redis统计在线,在SpringBoot中使用Redis的zset统计在线用户信息相关推荐

  1. SpringCloud学习笔记016---在windows下搭建的Redis服务_在SpringBoot中使用Redis

    1.首先搭建redis服务器,这个可以用centos,或者windows 2.我用centos搭建好了单机服务,但是连接的时候,就是报下面的错误,不理解:   没有解决 org.springframe ...

  2. redis基本操作和在springboot中的使用

    本文介绍redis的使用 redis启动步骤 说明 redis自增自减相关操作 redis string set操作 get操作 其他操作 redis hash set操作 get操作 其他操作 re ...

  3. springboot中使用redis详解

    一.redis简介 redis是一款高性能key-value(键值对)内存型数据库,是非关系型数据库的一种,它采用单线程的架构方式,避免了多线程存在的锁处理造成的资源耗费,读取速度非常快,非常适合变化 ...

  4. php执行who命令,Linux_在Linux系统中使用who和whoami命令获取用户信息,who用法:who [选项]... [ 文件 | - phpStudy...

    在Linux系统中使用who和whoami命令获取用户信息 who用法:who [选项]... [ 文件 | 参数1 参数2 ] 显示当前已登录的用户信息. -a, --all             ...

  5. java分布式会话redis_详解springboot中redis的使用和分布式session共享问题

    对于分布式使用Nginx+Tomcat实现负载均衡,最常用的均衡算法有IP_Hash.轮训.根据权重.随机等.不管对于哪一种负载均衡算法,由于Nginx对不同的请求分发到某一个Tomcat,Tomca ...

  6. SpringBoot中集成Redis实现对redis中数据的解析和存储

    场景 SpringBoot中操作spring redis的工具类: SpringBoot中操作spring redis的工具类_霸道流氓气质的博客-CSDN博客 上面讲的操作redis的工具类,但是对 ...

  7. SpringBoot中使用redis事务

    本文基于SpringBoot 2.X 事务在关系型数据库的开发中经常用到,其实非关系型数据库,比如redis也有对事务的支持,本文主要探讨在SpringBoot中如何使用redis事务. 事务的相关介 ...

  8. Docker中搭建redis分片集群,搭建redis哨兵结构,实现springboot中对redis分片集群、哨兵结构的访问,Redis缓存雪崩、缓存击穿处理(非关系型数据库技术课程 第十二周)

    文章目录 一.要求: 二.知识总结 缓存雪崩 解决方案 docker中redis分片集群搭建 配置好配置文件 redis-6380.conf redis-6381.conf redis-6382.co ...

  9. redis:01入门指南以及在springboot中使用redis

    https://redis.io/download step1:参考官网的安装很简单 wget http://download.redis.io/releases/redis-5.0.6.tar.gz ...

最新文章

  1. Dubbo zookeeper 分布式 集群问题
  2. Android之给图片添加水印效果
  3. 对用户输入内容进行字数提示功能
  4. 11.1自定义异常类
  5. 全球及中国手持式吸尘器行业供应需求及未来投资潜力预测报告2022-2027年
  6. 虚拟打印机开发日志(一):使用x64 WIN7编译环境编译的完整步骤
  7. c 引用mysql报错_安装TPCC-MySQL报错
  8. IOS(常用移动终端设备) push实现通知中心
  9. 介绍一个新鲜玩意 开源的杀毒软件
  10. Oracle游标使用
  11. (71)FPGA面试题-使用不同的代码实现2-4译码器?使用case语句
  12. Office组件无法正常使用的解决方法
  13. tcp下载窗口太小的问题_面试官:换人!他连 TCP 这几个参数都不懂(二)
  14. Oracle 11g R1(11.1) Joins表连接
  15. 轻松搞定C语言中复杂的声明
  16. svn中文语言安装包使用
  17. pandas训练集测试集划分_用pandas划分数据集实现训练集和测试集
  18. 声艺fx16调音台怎么样_声艺FX16II 声艺(Soundcraft) FX16ii 调音台
  19. 利用EditPlus制作Anki记忆卡批量导入文件
  20. DEC6713开发板的摸索(1)

热门文章

  1. IPRO_DOCXCC_EXTRACT_PARTBODY
  2. CRM Document builder fill-in status fillin
  3. IBASE 不能被编辑
  4. 在program A里访问program B local class的两种方式
  5. 在Kubernetes上运行SAP UI5应用
  6. SAP CRM里Lead通过工作流自动创建Opportunity的原理讲解
  7. 如何在WebIDE里打开一个HTML5应用
  8. msdn画圆弧函数_三角函数常识2020
  9. idea JDK安装与配置
  10. java字符串包ascii 方法amp;#39_用 Java 生成 ASCII 字符画