如何实现亿级用户在线状态统计?

亿级用户在线场景分析与解决方案

在这里插入图片描述

目录

  1. 亿级用户在线场景分析
  2. 解决方案
    • 2.1 基于总数的统计方案
    • 2.2 基于具体用户详情的统计方案
  3. 具体实现
    • 3.1 基于总数的统计方案
    • 3.2 基于用户标识的统计实现
    • 3.3 Spring Boot 中的实现
  4. 总结

1. 亿级用户在线场景分析

以 QQ 在线状态统计为例,其典型特征包括:数据量大、内存占用高、实时性要求高。传统的解决方案(如在数据库中为每个用户添加一个在线状态字段,上线设为 1,下线设为 0)在这种场景下显得力不从心。原因如下:

  • 数据库压力大:频繁的上、下线操作会导致数据库 IO 压力剧增。
  • 实时统计困难:频繁刷新查询会拖垮数据库性能,难以满足实时性要求。

因此,我们需要寻找更高效、更适合大规模场景的解决方案。


2. 解决方案

针对亿级用户在线状态统计,常见的解决方案可分为两类:

2.1 基于总数的统计方案

通过维护一个总在线人数计数器,用户上线时计数器加 1,下线时减 1,从而实现在线人数的统计。

优点
  • 实现简单,效率高。
  • 内存占用少。
缺点
  • 无法精确查询某个用户在某个时刻的在线状态。
  • 在异常退出应用的情况下,难以实现基于在线监测机制的重复下线判断。

2.2 基于具体用户详情的统计方案

将用户的标识(如 QQ 号)和在线状态存储在集合中,通过集合操作实现统计。

优点
  • 统计精准,可以查询某个用户在某个时刻的在线状态。
  • 在异常退出应用的情况下,可以精准地实现下线用户的去重功能。
缺点
  • 内存占用大。
  • 效率较低。

3. 具体实现

以下是两种方案的具体实现方式:

3.1 基于总数的统计方案

基于总数的统计可以通过以下两种方式实现:

3.1.1 基于 Redis 的 incrdecr 操作

使用 Redis 的 incr(加 1)和 decr(减 1)操作来维护在线人数计数器。用户上线时调用 incr,下线时调用 decr

3.1.2 基于 Redis 的 HyperLogLog

Redis 的 HyperLogLog(HLL)是一种高性能的基数(去重)统计数据结构,适用于大规模数据的去重统计。其优点是空间占用率极低(仅需 12KB 空间即可统计约 18 亿数据),但缺点是存在极低的误差率(约 0.81%)。HLL 的特点如下:

  • 无法移除元素。
  • 适合对误差容忍度较高的场景。

3.2 基于用户标识的统计实现

基于用户标识(如 QQ 号),可以使用 Redis 的 Bitmap(位数组) 来实现。Bitmap 的结构如下:

  • 每个下标表示一个具体的数字,值为 1 表示在线,值为 0 表示离线。
  • 例如,10 亿个数字占用的位数组空间为 10 亿 bit = 0.116 GB,空间占用量极小。
具体操作命令
  • 用户上线:使用 SETBIT 命令将对应位置设为 1。
  • 用户下线:使用 SETBIT 命令将对应位置设为 0。
  • 判断用户是否在线:使用 GETBIT 命令。
  • 统计在线用户数:使用 BITCOUNT 命令。

3.3 Spring Boot 中的实现

在 Spring Boot 项目中,可以使用 RedisTemplate 实现用户的上、下线设置以及在线人数统计。具体代码如下:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;@Service
public class BitmapService {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;/*** 设置 Bitmap 中的位* @param key 键* @param offset 偏移量* @param value 值(0 或 1)*/public void setBit(String key, long offset, boolean value) {redisTemplate.opsForValue().setBit(key, offset, value);}/*** 获取 Bitmap 中的位* @param key 键* @param offset 偏移量* @return 位的值(0 或 1)*/public boolean getBit(String key, long offset) {return redisTemplate.opsForValue().getBit(key, offset);}/*** 计算 Bitmap 中值为 1 的位的数量* @param key 键* @return 值为 1 的位的数量*/public Long bitCount(String key) {return redisTemplate.opsForValue().bitCount(key);}
}

4. 总结

在处理亿级用户在线状态统计时,选择合适的方案至关重要。基于总数的统计方案虽然简单高效,但缺乏精准性;而基于用户标识的统计方案虽然精准,但内存占用较大。结合实际需求,可以选择以下路径:
如果对实时性和性能要求极高,且可以容忍少量误差,可以选择基于 Redis 的 HyperLogLog 或 Bitmap 方案。
如果需要精准查询用户的在线状态,且对内存占用和效率要求较低,可以选择基于用户标识的集合方案。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/5435.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

多线程杂谈:惊群现象、CAS、安全的单例

引言 本文是一篇杂谈&#xff0c;帮助大家了解多线程可能会出现的面试题。 目录 引言 惊群现象 结合条件变量 CAS原子操作&#xff08;cmp & swap&#xff09; 线程控制&#xff1a;两个线程交替打印奇偶数 智能指针线程安全 单例模式线程安全 最简单的单例&…

sql实战解析-sum()over(partition by xx order by xx)

该窗口函数功能 sum( c )over( partition by a order by b) 按照一定规则汇总c的值&#xff0c;具体规则为以a分组&#xff0c;每组内按照b进行排序&#xff0c;汇总第一行至当前行的c的加和值。 从简单开始一步一步讲&#xff0c; 1、sum( )over( ) 对所有行进行求和 2、sum(…

你还在用idea吗

从VIM、Emacs&#xff0c;到eclipse、Jetbrains, 再到VSCode&#xff0c;过去的三十年时间&#xff0c;出现了这三代IDE产品。现在属于AI的时代来了&#xff0c;最新一代的产品像Cursor、Windsurf&#xff0c;就在昨天&#xff0c;字节跳动发布了最新的IDE&#xff0c;就叫Trae…

Unity新版InputSystem短按与长按,改键的实现

目录 前言&#xff1a; 一、InputSystem简介 1.安装InputSystem包 2.创建配置文件 3.创建自定义的Actions 二、自定义输入类 三、改键 四、全代码 前言&#xff1a; 新版inputsystem是Unity推出的一种新的输入方式&#xff0c;它将设备与行为进行分离&#xff0c;通过…

Android AutoMotive --CarService

1、AAOS概述 Android AutoMotive OS是谷歌针对车机使用场景打造的操作系统&#xff0c;它是基于现有Android系统的基础上增加了新特性&#xff0c;最主要的就是增加了CarService&#xff08;汽车服务&#xff09;模块。我们很容易把Android AutoMotive和Android Auto搞混&…

AWTK-WEB 快速入门(3) - C 语言 Http 应用程序

AWTK-WEB 快速入门 - C 语言 Http 应用程序 XMLHttpRequest 改变了 Web 应用程序与服务器交换数据的方式&#xff0c;fetch 是 XMLHttpRequest 继任者&#xff0c;具有更简洁的语法和更好的 Promise 集成。本文介绍一下如何使用 C 语言开发 AWTK-WEB 应用程序&#xff0c;并用 …

WPF1-从最简单的xaml开始

1. 最简单的WPF应用 1.1. App.config1.2. App.xaml 和 App.xaml.cs1.3. MainWindow.xaml 和 MainWindow.xaml.cs 2. 正式开始分析 2.1. 声明即定义2.2. 命名空间 2.2.1. xaml的Property和Attribute2.2.2. xaml中命名空间2.2.3. partial关键字 学习WPF&#xff0c;肯定要先学…

C#与AI的共同发展

C#与人工智能(AI)的共同发展反映了编程语言随着技术进步而演变&#xff0c;以适应新的挑战和需要。自2000年微软推出C#以来&#xff0c;这门语言经历了多次迭代&#xff0c;不仅成为了.NET平台的主要编程语言之一&#xff0c;还逐渐成为构建各种类型应用程序的强大工具。随着时…

图解Git——分布式Git《Pro Git》

分布式工作流程 Centralized Workflow&#xff08;集中式工作流&#xff09; 所有开发者都与同一个中央仓库同步代码&#xff0c;每个人通过拉取、提交来合作。如果两个开发者同时修改了相同的文件&#xff0c;后一个开发者必须在推送之前合并其他人的更改。 Integration-Mana…

2025年最新汽车零部件企业销售项目管理解决方案

在汽车零部件企业&#xff0c;销售项目管理的不规范和销售预测的不准确性常导致生产计划无法及时调整&#xff0c;因此客户关系常常中断&#xff0c;导致企业业务机会的丧失。为解决该问题&#xff0c;企业需要投入更多资源以优化销售流程与销售预测。 1、360多维立体客户视图…

vscode导入模块不显示类型注解

目录结构&#xff1a; utils.py&#xff1a; import random def select_Jrandom(i:int, m:int) -> int:"""随机选择一个不等于 i 的整数"""j iwhile j i:j int(random.uniform(0, m))return jdef clip_alpha(alpha_j:float, H:float, L:f…

【Elasticsearch】 Ingest Pipeline `processors`属性详解

在Elasticsearch中&#xff0c;Ingest Pipeline 的 processors 属性是一个数组&#xff0c;包含一个或多个处理器&#xff08;processors&#xff09;。每个处理器定义了一个数据处理步骤&#xff0c;可以在数据索引之前对数据进行预处理或富化。以下是对 processors 属性中常见…

python转转商超书籍信息爬虫

1基本理论 1.1概念体系 网络爬虫又称网络蜘蛛、网络蚂蚁、网络机器人等&#xff0c;可以按照我们设置的规则自动化爬取网络上的信息&#xff0c;这些规则被称为爬虫算法。是一种自动化程序&#xff0c;用于从互联网上抓取数据。爬虫通过模拟浏览器的行为&#xff0c;访问网页并…

Ext2 文件系统:数字世界的基石,深度解码超时空存储魔法

本篇博主将带大家深入底层探秘系统是如何与磁盘进行相互交流的&#xff0c;配合精美配图&#xff0c;细节讲解来带大家深入探究&#xff08;注&#xff1a;本篇文章建议了解磁盘内部物理结果组成及设计再进行阅读&#xff09;。 羑悻的小杀马特.-CSDN博客羑悻的小杀马特.擅长C…

postman的使用

Postman是Restful API的测试工具。简单来讲是一款支持http协议的接口调试与测试工具&#xff0c;其主要特点就是功能强大、使用简单。通常无论是开发人员进行接口调试&#xff0c;还是测试人员做接口测试&#xff0c;postman通常都是首选工具。 注&#xff1a;作为开发人员对于…

模块化架构与微服务架构,哪种更适合桌面软件开发?

前言 在现代软件开发中&#xff0c;架构设计扮演着至关重要的角色。两种常见的架构设计方法是模块化架构与微服务架构。它们各自有独特的优势和适用场景&#xff0c;尤其在C#桌面软件开发领域&#xff0c;模块化架构往往更加具有实践性。本文将对这两种架构进行对比&#xff0…

工程上LabVIEW常用的控制算法有哪些

在工程应用中&#xff0c;LabVIEW常用的控制算法有很多&#xff0c;它们广泛应用于自动化、过程控制、机器人、测试测量等领域。以下是一些常见的控制算法&#xff1a; 1. PID 控制 用途&#xff1a;PID&#xff08;比例-积分-微分&#xff09;控制是最常用的反馈控制算法&…

nuxt3项目打包部署到服务器后配置端口号和开启https

nuxt3打包后的项目部署相对于一般vite打包的静态文件部署要稍微麻烦一些&#xff0c;还有一个主要的问题是开发环境配置的.env环境变量在打包后部署时获取不到&#xff0c;具体的解决方案可以参考我之前文章 nuxt3项目打包后获取.env设置的环境变量无效的解决办法。 这里使用的…

ui文件转py程序的工具

源博客连接&#xff1a; PyCharm中利用外部工具uic转成的py文件&#xff0c;里面全是C代码&#xff0c;并非python类型的代码&#xff0c;导致大量报错。。。_pyside6-uic为什么把ui转为了c-CSDN博客 如果想把ui文件转为py文件&#xff0c;首先设置pycharm的外部工具&#xf…

c++学习第七天

创作过程中难免有不足&#xff0c;若您发现本文内容有误&#xff0c;恳请不吝赐教。 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考。 一、const成员函数 //Date.h#pragma once#include<iostream> using namespace std;class Date { public:Date…