gpt4 book ai didi

MySQL去除重叠时间求时间差和的实现

转载 作者:qq735679552 更新时间:2022-09-29 22:32:09 25 4
gpt4 key购买 nike

CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.

这篇CFSDN的博客文章MySQL去除重叠时间求时间差和的实现由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.

我个人并不推荐在实际开发中使用存储过程,充满了各种的不方便,之所以写这东西,全在于学习,如果有高手看到我的内容有问题,可以随时指出或向我开炮.

需求:

在生产中常常出现计算两个时间差的业务,比如总宕机时间、总开通会员时间等等。。。但是这些时间往往不是连贯的,断断续续,甚至可能会出现重叠的情况。无法直接求出时间差.

例如:

MySQL去除重叠时间求时间差和的实现

MySQL去除重叠时间求时间差和的实现

开车:

一开始,我想的是用单条SQL实现,例如:↓ 。

  1. SELECT TIMESTAMPDIFF(MINUTE, '2021-08-19 14:30:00', '2021-08-19 15:00:00') FROM DUAL;

我发现,数据库数据千千万,不可能这样,也不可能用UNION这种东西去拼接,数据很多,就一定会有循环,所以,在不使用Java语言的情况下,我选择尝试用存储过程来解决以下这个问题.

思路:

首先,一次进入循环的数据不会进行计算,防止后边的数据和它有重叠, 。

从第二条数据开始,就要判断开始时间是否和上一个数据重叠,如果重叠,则校验结束时间是否也重叠,如果重叠我就啥也不干,不重叠,则把这个值赋给上一次的数据的结束时间.

如果开始时间不再范围内,那么需要判断开始时间是在上一次时间的之前还是之后 。

如果这个范围之前,把这个值赋给上一次的数据的开始时间.

在这个范围之后,计算并赋值 。

最后一次循环也要计算并赋值 。

实现:

首先创建表,模拟数据 。

  1. CREATE TABLE test01 (
  2. id int(32) unsigned NOT NULL AUTO_INCREMENT,
  3. start_time datetime NOT NULL,
  4. end_time datetime NOT NULL,
  5. PRIMARY KEY (`id`)
  6. )
  7.  
  8. INSERT INTO test01(id, start_time, end_time) VALUES (1, '2021-08-18 16:27:51', '2021-08-18 17:27:59');
  9. INSERT INTO test01(id, start_time, end_time) VALUES (2, '2021-08-18 17:20:26', '2021-08-18 20:10:37');
  10. INSERT INTO test01(id, start_time, end_time) VALUES (3, '2021-08-18 22:05:57', '2021-08-18 23:55:20');

MySQL去除重叠时间求时间差和的实现

创建存储过程:

  1. CREATE PROCEDURE sumTime()
  2. BEGIN
  3. -- 定义变量
  4.  
  5. -- 是否首次
  6. DECLARE is_old int(1) DEFAULT 0;
  7.  
  8. -- 上一次数据
  9. DECLARE old_start_time datetime;
  10. DECLARE old_end_time datetime;
  11.  
  12. -- 本次数据
  13. DECLARE start_time datetime;
  14. DECLARE end_time datetime;
  15.  
  16. -- 返回结果
  17. DECLARE num int(32) DEFAULT 0;
  18.  
  19. -- 循环结束开关
  20. DECLARE done int DEFAULT 0;
  21.  
  22. -- 创建游标(查询数据库数据)
  23. DECLARE list CURSOR FOR SELECT a.start_time, a.end_time FROM test01 a;
  24.  
  25. -- 定义最后一次循环时设置 循环结束开关 1
  26. DECLARE CONTINUE HANDLER FOR SQLSTATE '02000' SET done=1;
  27.  
  28. -- 开启游标
  29. OPEN list;
  30.  
  31. -- 开启循环
  32. posLoop:LOOP
  33. -- 取值 将当前循环的值取出 赋值给当前数据变量
  34. FETCH list INTO start_time,end_time;
  35. -- 判断是否首次
  36. if (is_old = 0) THEN
  37.  
  38. SET is_old = 1;
  39. SET old_start_time = start_time;
  40. SET old_end_time = end_time;
  41.  
  42. -- 否则
  43. ELSE
  44. -- 校验是否在区间内
  45. if (start_time >= old_start_time AND start_time <= old_end_time) THEN
  46.  
  47. -- 校验结束时间是否不在在区间内
  48. if (end_time < old_start_time OR end_time > old_end_time) THEN
  49. SET old_end_time = end_time;
  50. END IF;
  51.  
  52. -- 否则
  53. ELSE
  54.  
  55. if (start_time < old_start_time ) THEN
  56.  
  57. SET old_start_time = start_time;
  58.  
  59. ELSE
  60.  
  61. SET num = num + TIMESTAMPDIFF(MINUTE, old_start_time, old_end_time);
  62. SET old_start_time = start_time;
  63. SET old_end_time = end_time;
  64. END IF;
  65. END IF;
  66. END IF;
  67. -- 校验是否最后一次循环
  68. IF done=1 THEN
  69. SET num = num + TIMESTAMPDIFF(MINUTE, old_start_time, old_end_time);
  70. LEAVE posLoop;
  71. END IF;
  72. -- 结束循环
  73. END LOOP posLoop;
  74. -- 关闭游标
  75. CLOSE list;
  76. SELECT num;
  77. END;
  1. -- 调用存储过程
  2. call sumTime();

MySQL去除重叠时间求时间差和的实现

  1. -- 删除存储过程
  2. drop procedure if exists sumTime;

到此这篇关于MySQL去除重叠时间求时间差和的实现的文章就介绍到这了,更多相关MySQL 求时间差和内容请搜索我以前的文章或继续浏览下面的相关文章希望大家以后多多支持我! 。

原文链接:https://blog.csdn.net/qq_33211189/article/details/119823541 。

最后此篇关于MySQL去除重叠时间求时间差和的实现的文章就讲到这里了,如果你想了解更多关于MySQL去除重叠时间求时间差和的实现的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com