深入解析:如何优雅计算时间区间内的有效时长

前言

最近有一个需求是这样的,需要统计老师在教室内的实际时长,目前,客户端会每隔1分钟上报一次状态,并将上报数据存入数据库中。 我们需要计算在特定时间区间内的有效持续时间。本文将深入分析一个复杂但高效的SQL解决方案,该方案可以计算在指定时间段内的有效在线时长。

核心SQL解析

sql 复制代码
SELECT COALESCE(
         GREATEST(UNIX_TIMESTAMP(MIN(created_at)) - UNIX_TIMESTAMP(#{startTime}), 0) +
         COALESCE(
           SUM(UNIX_TIMESTAMP(next_time) - UNIX_TIMESTAMP(created_at)), 0
         ) +
         GREATEST(UNIX_TIMESTAMP(#{endTime}) - UNIX_TIMESTAMP(MAX(created_at)), 0
       , 0) AS total_seconds
FROM (
  SELECT
    n1.created_at,
    (
      SELECT MIN(n2.created_at)
      FROM network_stats n2
      WHERE n2.klass_id = n1.klass_id
        AND n2.server = n1.server
        AND n2.created_at > n1.created_at
    ) AS next_time
  FROM network_stats n1
  WHERE n1.klass_id = #{klassId}
    AND n1.server = #{server}
    AND n1.created_at >= #{startTime}
) t

关键技术点

时间区间处理的三段式模型

  • 起始段处理: GREATEST(UNIX_TIMESTAMP(MIN(created_at)) - UNIX_TIMESTAMP(#{startTime}), 0)

    • 计算第一条记录与查询开始时间的时间差
    • GREATEST可以确保结果不为负
  • 中间段处理: SUM(UNIX_TIMESTAMP(next_time) - UNIX_TIMESTAMP(created_at))

    • 计算所有相邻记录之间的时间差总和
    • 统计系统正常运行的累计时长
  • 结束段处理: GREATEST(UNIX_TIMESTAMP(#{endTime}) - UNIX_TIMESTAMP(MAX(created_at)), 0)

    • 计算最后一条记录与查询结束时间的时间差
    • 同样使用GREATEST避免负值

子查询与自连接技巧

ini 复制代码
SELECT MIN(n2.created_at)
FROM network_stats n2
WHERE n2.klass_id = n1.klass_id
  AND n2.server = n1.server
  AND n2.created_at > n1.created_at

这段sql在整段sql中属于内部子查询,使用了巧妙的自我引用技术

实现效果:

  • 对每条记录n1,查找同一klass_id和server的下一条记录n2
  • 构建了记录链,使每条记录都知道自己的"下一跳"时间

防御性编程思想

SQL中多处体现了防御性编程:

  • COALESCE处理可能的NULL值
  • GREATEST防止时间计算出现负数
COALESCE函数:数据安全的守护者

定义与语法:

scss 复制代码
COALESCE(expression1, expression2, ..., expressionN)

核心作用:

  • 按顺序检查参数,返回第一个非NULL的值
  • 相当于其他编程语言中的"空值合并运算符"

在本SQL中的应用:

scss 复制代码
COALESCE(SUM(UNIX_TIMESTAMP(next_time) - UNIX_TIMESTAMP(created_at)), 0)
  • 当SUM结果为NULL时(无符合条件的数据),返回0
  • 避免NULL参与后续计算导致整个表达式结果为NULL

实际业务意义:

  • 确保没有数据记录时仍能返回合理的0值
  • 防止前端应用因收到NULL而崩溃
GREATEST函数:智能的边界守卫

定义与语法:

scss 复制代码
GREATEST(value1, value2, ..., valueN)

核心作用:

  • 返回参数列表中的最大值
  • 常用于确保数值不低于某个阈值

在本SQL中的应用:

scss 复制代码
GREATEST(UNIX_TIMESTAMP(MIN(created_at)) - UNIX_TIMESTAMP(#{startTime}), 0)
  • 计算第一条记录与查询开始时间的时间差
  • 当MIN(created_at)早于startTime时,避免出现负值

实际业务意义:

  • 保证时间差值不会为负,符合业务逻辑(时长不能为负)
  • 处理数据早于查询时间范围的边界情况

延伸应用场景

这种计算模式还可以适用于:

  • 服务可用性监控 :计算服务器在特定时段内的实际在线时长
  • 用户行为分析 :统计用户在APP中的有效停留时间
  • 设备监控 :计算设备在线时长用于计费或维护

注意

索引优化: 确保(klass_id, server, created_at)有复合索引

最后

对了,我们的项目中使用的是MySQL 5.7MySQL 8.0应该可以使用窗口函数LEAD来实现。

相关推荐
可乐ea12 分钟前
【Spring Boot + MyBatis|第7篇】JWT 登录认证与拦截器实现
java·spring boot·后端·mybatis·状态模式
加加and减减25 分钟前
Docker真实安装mysql8教程并优化配置
运维·mysql·docker·容器
西安邮电大学31 分钟前
有关栈的经典算法题
java·后端·其他·算法·面试
程序猿乐锅40 分钟前
【MySQL | 第九篇】MySQL 存储过程
数据库·mysql
摇滚侠1 小时前
SpringMVC 入门到实战 配置类替换 XML 配置文件 86-91
xml·java·后端·spring·maven·intellij-idea
我登哥MVP1 小时前
SpringCloud Alibaba 核心组件解析:服务注册与发现(Nacos)
java·spring boot·后端·spring·spring cloud·java-ee·maven
王小王-1231 小时前
基于深度学习的个性化音乐推荐系统的设计与开发
人工智能·深度学习·mysql·vue·推荐算法·个性化音乐推荐系统·音乐预测
摇滚侠1 小时前
SpringMVC 入门到实战 处理静态资源的过程 64
java·后端·spring·maven·intellij-idea
Liquad Li1 小时前
ABP vNext 标准分层解决方案项目结构完整解析
后端
学代码的真由酱2 小时前
【自用】接口测试
接口测试·postman·测试·cookie·token鉴权