经典sql题(七)查找直播间最大在线人数

使用 SQL 分析房间用户状态变化

本文将详细介绍如何使用 SQL 的窗口函数和聚合函数,分析用户在房间中的状态变化,目标是计算每个房间指定时间段内的最大用户状态。

示例数据

假设我们的数据表包含以下字段:

room_id user_id login_time logout_time
101 1 2023-03-10 12:05:00 2023-03-10 12:45:00
101 2 2023-03-10 12:10:00 2023-03-10 12:50:00
102 3 2023-03-10 12:00:00 2023-03-10 13:00:00
102 4 2023-03-10 12:30:00 2023-03-10 13:30:00

第一步:提取事件并标记类型

在这一步,我们将从原始数据中提取每个用户的登录和退出事件,并为每个事件分配一个类型标记(1 表示登录,-1 表示退出)。

提取登录事件
sql 复制代码
SELECT 
    room_id,
    user_id,
    UNIX_TIMESTAMP(login_time) AS event_time,
    1 AS user_type
FROM 
    table
WHERE 
    FROM_UNIXTIME(UNIX_TIMESTAMP(login_time), '%Y%m%d') = '20230310'
    AND (HOUR(login_time) BETWEEN 12 AND 13)
结果示例
room_id user_id event_time user_type
101 1 1615375500 1
101 2 1615375800 1
102 3 1615375200 1
102 4 1615377000 1
提取退出事件
sql 复制代码
SELECT 
    room_id,
    user_id,
    UNIX_TIMESTAMP(logout_time) AS event_time,
    -1 AS user_type
FROM 
    table
WHERE 
    FROM_UNIXTIME(UNIX_TIMESTAMP(logout_time), '%Y%m%d') = '20230310'
    AND (HOUR(logout_time) BETWEEN 12 AND 13)
结果示例
room_id user_id event_time user_type
101 1 1615377900 -1
101 2 1615378200 -1
102 3 1615378800 -1
102 4 1615379400 -1
合并结果

我们将登录和退出事件通过 UNION ALL 合并,以获得完整的事件列表。

sql 复制代码
SELECT 
    room_id,
    user_id,
    event_time,
    user_type
FROM (
    -- 登录事件
    SELECT 
        room_id,
        user_id,
        UNIX_TIMESTAMP(login_time) AS event_time,
        1 AS user_type
    FROM 
        table
    WHERE 
        FROM_UNIXTIME(UNIX_TIMESTAMP(login_time), '%Y%m%d') = '20230310'
        AND (HOUR(login_time) BETWEEN 12 AND 13)
    UNION ALL
    -- 退出事件
    SELECT 
        room_id,
        user_id,
        UNIX_TIMESTAMP(logout_time) AS event_time,
        -1 AS user_type
    FROM 
        table
    WHERE 
        FROM_UNIXTIME(UNIX_TIMESTAMP(logout_time), '%Y%m%d') = '20230310'
        AND (HOUR(logout_time) BETWEEN 12 AND 13)
) AS events
合并结果示例
room_id user_id event_time user_type
101 1 1615375500 1
101 2 1615375800 1
102 3 1615375200 1
102 4 1615377000 1
101 1 1615377900 -1
101 2 1615378200 -1
102 3 1615378800 -1
102 4 1615379400 -1

第二步:计算累积用户状态

使用窗口函数累积计算用户状态。

sql 复制代码
SELECT
    room_id,
    user_id,
    user_type,
    SUM(user_type) OVER (PARTITION BY room_id ORDER BY event_time) AS status
FROM (
    -- 上一步的查询
) AS events

结果示例

room_id user_id user_type status
101 1 1 1
101 2 1 2
101 1 -1 1
101 2 -1 0
102 3 1 1
102 4 1 2
102 3 -1 1
102 4 -1 0

第三步:求最大状态

通过聚合函数求每个房间的最大状态。

sql 复制代码
SELECT
    room_id,
    MAX(status) AS max_status
FROM (
    -- 上一步的状态查询
) AS status_calculation
GROUP BY
    room_id;

结果示例

room_id max_status
101 2
102 2

解析

  • 数据准备:合并登录和退出事件,标记事件类型。
  • 计算状态 :使用 SUM() OVER 计算累积状态。
  • 求最大状态 :通过 MAX(status) 获取最大用户状态。
相关推荐
乾元10 小时前
如何把 CCIE / HCIE 的实验案例改造成 AI 驱动的工程项目——从“实验室能力”到“可交付系统”的完整迁移路径
大数据·运维·网络·人工智能·深度学习·安全·机器学习
爱写bug的野原新之助10 小时前
数据库及navicat工具
数据库·网络爬虫·工具
数据知道10 小时前
一文掌握 MongoDB 存储引擎 WiredTiger 的原理
数据库·mongodb·数据库架构
xiaobaishuoAI10 小时前
后端工程化实战指南:从规范到自动化,打造高效协作体系
java·大数据·运维·人工智能·maven·devops·geo
Full Stack Developme10 小时前
Mycat 2 实现 MySQL 读写分离,并且实现 主从同步
android·数据库·mysql
我是人✓10 小时前
Spring IOC入门
java·数据库·spring
Hello.Reader10 小时前
PyFlink DataStream 程序骨架、常用 Source/Sink、状态(State)、与 Table/SQL 互转一篇搞定
数据库·sql·linq
俊哥大数据10 小时前
【实战项目5】基于Flink新闻热搜大数据实时分析项目
大数据·flink
俊哥大数据10 小时前
【实战项目3】基于Flink广告投放业务领域大数据实时分析项目
大数据·flink
三不原则10 小时前
故障案例:模型推理响应慢,排查 Redis 缓存集群问题
数据库·redis·缓存