草庐IT

mysql - 计算 MySQL 中时间重叠的重叠持续时间

coder 2023-10-22 原文

我有一个包含事件的数据库表:

CREATE TABLE events
(event VARCHAR(32)
,down_time TIMESTAMP
,up_time TIMESTAMP
,id INT UNSIGNED NOT NULL AUTO_INCREMENT KEY
,INDEX(event(16))
);

INSERT INTO events(event, down_time, up_time) VALUES
('e1', '2015-01-01 00:00:03', '2015-01-01 00:00:08'),
('e2', '2015-01-01 00:00:05', '2015-01-01 00:00:06'),
('e3', '2015-01-01 00:00:02', '2015-01-01 00:00:09'),
('e4', '2015-01-01 00:00:01', '2015-01-01 00:00:04'),
('e5', '2015-01-01 00:00:07', '2015-01-01 00:00:10');

select * from events;
| event | down_time           | up_time             | id |
+-------+---------------------+---------------------+----+
| e1    | 2015-01-01 00:00:03 | 2015-01-01 00:00:08 |  1 |
| e2    | 2015-01-01 00:00:05 | 2015-01-01 00:00:06 |  2 |
| e3    | 2015-01-01 00:00:02 | 2015-01-01 00:00:09 |  3 |
| e4    | 2015-01-01 00:00:01 | 2015-01-01 00:00:04 |  4 |
| e5    | 2015-01-01 00:00:07 | 2015-01-01 00:00:10 |  5 |

我使用以下查询 (query1) 发现事件重叠:

SELECT * 
  FROM events a 
  JOIN events b 
    ON a.down_time <= b.up_time 
   AND a.up_time >= b.down_time 
 WHERE a.id < b.id 
   AND a.event != b.event 
   AND a.event regexp 'e[1-5]' 
   AND b.event regexp 'e[1-5]';

产生以下事件重叠(结果 1):

| event | down_time           | up_time             | id | event | down_time           | up_time             | id |
+-------+---------------------+---------------------+----+-------+---------------------+---------------------+----+
| e1    | 2015-01-01 00:00:03 | 2015-01-01 00:00:08 |  1 | e2    | 2015-01-01 00:00:05 | 2015-01-01 00:00:06 |  2 |
| e1    | 2015-01-01 00:00:03 | 2015-01-01 00:00:08 |  1 | e3    | 2015-01-01 00:00:02 | 2015-01-01 00:00:09 |  3 |
| e1    | 2015-01-01 00:00:03 | 2015-01-01 00:00:08 |  1 | e4    | 2015-01-01 00:00:01 | 2015-01-01 00:00:04 |  4 |
| e1    | 2015-01-01 00:00:03 | 2015-01-01 00:00:08 |  1 | e5    | 2015-01-01 00:00:07 | 2015-01-01 00:00:10 |  5 |
| e2    | 2015-01-01 00:00:05 | 2015-01-01 00:00:06 |  2 | e3    | 2015-01-01 00:00:02 | 2015-01-01 00:00:09 |  3 |
| e3    | 2015-01-01 00:00:02 | 2015-01-01 00:00:09 |  3 | e4    | 2015-01-01 00:00:01 | 2015-01-01 00:00:04 |  4 |
| e3    | 2015-01-01 00:00:02 | 2015-01-01 00:00:09 |  3 | e5    | 2015-01-01 00:00:07 | 2015-01-01 00:00:10 |  5 |
+-------+---------------------+---------------------+----+-------+---------------------+---------------------+----+

我想在事件重叠 (results1) 中显示每一行的重叠持续时间,目前在 PHP 中使用以下条件测试:

    if (a.down_time <= b.down_time && b.up_time <= a.up_time)
    {
    overlap_duration = b.up_time-b.down_time;
    }
    else if (a.down_time >= b.down_time && a.up_time <= b.up_time)
    {
    overlap_duration = a.up_time-a.down_time;
    }
    else if (a.down_time <= b.down_time)
    {
    overlap_duration = a.up_time-b.down_time;
    }
    else if (a.down_time >= b.down_time)
    {
    overlap_duration = b.up_time-a.down_time;
    }
1   2   3   4   5   6   7   8   9   10
|   |   |   |   |   |   |   |   |   |
|   |   |----------e1-------|   |   |
|   |   |   |   | e2|   |   |   |   | a.down_time <= b.down_time && b.up_time <= a.up_time
|   |------------e3-------------|   | a.down_time >= b.down_time && a.up_time <= b.up_time
|----e4-----|   |   |   |   |   |   | a.down_time >= b.down_time
|   |   |   |   |   |   |-----e5----| a.down_time <= b.down_time

然后产生以下输出(results2):

|  event    | down_time            | up_time              | duration    |
|  e1       | 2015-01-01 00:00:03    2015-01-01 00:00:08    00:00:00:05 |
|  e2       | 2015-01-01 00:00:05    2015-01-01 00:00:06    00:00:00:01 |
|  Overlap1 | 2015-01-01 00:00:05    2015-01-01 00:00:06    00:00:00:01 |
|           |                                                           |
|  e1       | 2015-01-01 00:00:03    2015-01-01 00:00:08    00:00:00:05 |
|  e3       | 2015-01-01 00:00:02    2015-01-01 00:00:09    00:00:00:07 |
|  Overlap2 | 2015-01-01 00:00:03    2015-01-01 00:00:08    00:00:00:05 |
|           |                                                           |
|  e1       | 2015-01-01 00:00:03    2015-01-01 00:00:08    00:00:00:05 |
|  e4       | 2015-01-01 00:00:01    2015-01-01 00:00:04    00:00:00:03 |
|  Overlap3 | 2015-01-01 00:00:03    2015-01-01 00:00:04    00:00:00:01 |
|           |                                                           |
|  e1       | 2015-01-01 00:00:03    2015-01-01 00:00:08    00:00:00:05 |
|  e5       | 2015-01-01 00:00:07    2015-01-01 00:00:10    00:00:00:03 |
|  Overlap4 | 2015-01-01 00:00:07    2015-01-01 00:00:08    00:00:00:01 |
|           |                                                           |
|  e2       | 2015-01-01 00:00:05    2015-01-01 00:00:06    00:00:00:01 |
|  e3       | 2015-01-01 00:00:02    2015-01-01 00:00:09    00:00:00:07 |
|  Overlap5 | 2015-01-01 00:00:05    2015-01-01 00:00:06    00:00:00:01 |
|           |                                                           |
|  e3       | 2015-01-01 00:00:02    2015-01-01 00:00:09    00:00:00:07 |
|  e4       | 2015-01-01 00:00:01    2015-01-01 00:00:04    00:00:00:03 |
|  Overlap6 | 2015-01-01 00:00:02    2015-01-01 00:00:04    00:00:00:02 |
|           |                                                           |
|  e3       | 2015-01-01 00:00:02    2015-01-01 00:00:09    00:00:00:07 |
|  e5       | 2015-01-01 00:00:07    2015-01-01 00:00:10    00:00:00:03 |
|  Overlap7 | 2015-01-01 00:00:07    2015-01-01 00:00:09    00:00:00:02 |
+-----------+-----------------------------------------------------------+

我怀疑通过处理 results1 产生 results2 来计算 MySQL 中的重叠持续时间可能会更好 但不确定如何最好地进行……如果有的话?提前致谢。

最佳答案

这应该按预期工作

SELECT
    a.*,
    b.*,
    (least(a.up_time, b.up_time) - greatest(a.down_time, b.down_time)) as overlap_seconds
  FROM events a 
  JOIN events b 
    ON a.down_time <= b.up_time 
   AND a.up_time >= b.down_time 
 WHERE a.id < b.id 
   AND a.event != b.event
   AND a.event regexp 'e[1-5]' 
   AND b.event regexp 'e[1-5]'
GROUP BY a.id, b.id;

要获得重叠,您需要做的就是将每一行的“最小”正常运行时间与“最大”停机时间进行比较 ...

关于mysql - 计算 MySQL 中时间重叠的重叠持续时间,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32749274/

有关mysql - 计算 MySQL 中时间重叠的重叠持续时间的更多相关文章

  1. ruby-on-rails - 使用一系列等级计算字母等级 - 2

    这里是Ruby新手。完成一些练习后碰壁了。练习:计算一系列成绩的字母等级创建一个方法get_grade来接受测试分数数组。数组中的每个分数应介于0和100之间,其中100是最大分数。计算平均分并将字母等级作为字符串返回,即“A”、“B”、“C”、“D”、“E”或“F”。我一直返回错误:avg.rb:1:syntaxerror,unexpectedtLBRACK,expecting')'defget_grade([100,90,80])^avg.rb:1:syntaxerror,unexpected')',expecting$end这是我目前所拥有的。我想坚持使用下面的方法或.join,

  2. ruby-on-rails - Ruby 检查日期时间是否为 iso8601 并保存 - 2

    我需要检查DateTime是否采用有效的ISO8601格式。喜欢:#iso8601?我检查了ruby​​是否有特定方法,但没有找到。目前我正在使用date.iso8601==date来检查这个。有什么好的方法吗?编辑解释我的环境,并改变问题的范围。因此,我的项目将使用jsapiFullCalendar,这就是我需要iso8601字符串格式的原因。我想知道更好或正确的方法是什么,以正确的格式将日期保存在数据库中,或者让ActiveRecord完成它们的工作并在我需要时间信息时对其进行操作。 最佳答案 我不太明白你的问题。我假设您想检查

  3. ruby-on-rails - 将 Ruby 中的日期/时间格式化为 YYYY-MM-DD HH :MM:SS - 2

    这个问题在这里已经有了答案:Railsformattingdate(4个答案)关闭4年前。我想格式化Time.Now函数以显示YYYY-MM-DDHH:MM:SS而不是:“2018-03-0909:47:19+0000”该函数需要放在时间中.现在功能。require‘roo’require‘roo-xls’require‘byebug’file_name=ARGV.first||“Template.xlsx”excel_file=Roo::Spreadsheet.open(“./#{file_name}“,extension::xlsx)xml=Nokogiri::XML::Build

  4. ruby - 查找字符串中的内容类型(数字、日期、时间、字符串等) - 2

    我正在尝试解析一个CSV文件并使用SQL命令自动为其创建一个表。CSV中的第一行给出了列标题。但我需要推断每个列的类型。Ruby中是否有任何函数可以找到每个字段中内容的类型。例如,CSV行:"12012","Test","1233.22","12:21:22","10/10/2009"应该产生像这样的类型['integer','string','float','time','date']谢谢! 最佳答案 require'time'defto_something(str)if(num=Integer(str)rescueFloat(s

  5. 使用canal同步MySQL数据到ES - 2

    文章目录一、概述简介原理模块二、配置Mysql使用版本环境要求1.操作系统2.mysql要求三、配置canal-server离线下载在线下载上传解压修改配置单机配置集群配置分库分表配置1.修改全局配置2.实例配置垂直分库水平分库3.修改group-instance.xml4.启动监听四、配置canal-adapter1修改启动配置2配置映射文件3启动ES数据同步查询所有订阅同步数据同步开关启动4.验证五、配置canal-admin一、概述简介canal是Alibaba旗下的一款开源项目,Java开发。基于数据库增量日志解析,提供增量数据订阅&消费。Git地址:https://github.co

  6. jenkins部署1--jenkins+gitee持续集成 - 2

    前置步骤我们都操作完了,这篇开始介绍jenkins的集成。话不多说,看操作1、登录进入jenkins后会让你选择安装插件,选择第一个默认的就行。安装完成后设置账号密码,重新登录。2、配置JDK和Git都需要执行路径,所以需要先把执行路径找到,先进入服务器的docker容器,2.1JDK的路径root@69eef9ee86cf:/usr/bin#echo$JAVA_HOME/usr/local/openjdk-82.2Git的路径root@69eef9ee86cf:/#whichgit/usr/bin/git3、先配置JDK和Git。点击:ManageJenkins>>GlobalToolCon

  7. 计算机毕业设计ssm+vue基本微信小程序的小学生兴趣延时班预约小程序 - 2

    项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU

  8. sql - 查询忽略时间戳日期的时间范围 - 2

    我正在尝试查询我的Rails数据库(Postgres)中的购买表,我想查询时间范围。例如,我想知道在所有日期的下午2点到3点之间进行了多少次购买。此表中有一个created_at列,但我不知道如何在不搜索特定日期的情况下完成此操作。我试过:Purchases.where("created_atBETWEEN?and?",Time.now-1.hour,Time.now)但这最终只会搜索今天与那些时间的日期。 最佳答案 您需要使用PostgreSQL'sdate_part/extractfunction从created_at中提取小时

  9. ruby - 如何计算 Liquid 中的变量 +1 - 2

    我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我

  10. ruby - 使用 Ruby,计算 n x m 数组的每一列中有多少个 true 的简单方法是什么? - 2

    给定一个nxmbool数组:[[true,true,false],[false,true,true],[false,true,true]]有什么简单的方法可以返回“该列中有多少个true?”结果应该是[1,3,2] 最佳答案 使用转置得到一个数组,其中每个子数组代表一列,然后将每一列映射到其中的true数:arr.transpose.map{|subarr|subarr.count(true)}这是一个带有inject的版本,应该在1.8.6上运行,没有任何依赖:arr.transpose.map{|subarr|subarr.in

随机推荐