当前位置: 首页 > article >正文

【MySQL】ROW_NUMBER 窗口函数妙用之报告系统状态的连续日期

力扣题

1、题目地址

1225. 报告系统状态的连续日期

2、模拟表

表:Failed

Column NameType
fail_datedate
  • 该表主键为 fail_date (具有唯一值的列)。
  • 该表包含失败任务的天数.

表: Succeeded

Column NameType
success_datedate
  • 该表主键为 success_date (具有唯一值的列)。
  • 该表包含成功任务的天数.

3、要求

系统 每天 运行一个任务。每个任务都独立于先前的任务。任务的状态可以是失败或是成功。

编写解决方案找出 2019-01-01 到 2019-12-31 期间任务连续同状态 period_state 的起止日期(start_date 和 end_date)。

即如果任务失败了,就是失败状态的起止日期,如果任务成功了,就是成功状态的起止日期。

最后结果按照起始日期 start_date 排序。

4、示例

输入:
Failed 表:

fail_date
2018-12-28
2018-12-29
2019-01-04
2019-01-05

Succeeded 表:

success_date
2018-12-30
2018-12-31
2019-01-01
2019-01-02
2019-01-03
2019-01-06

输出:

period_statestart_dateend_date
succeeded2019-01-012019-01-03
failed2019-01-042019-01-05
succeeded2019-01-062019-01-06

解释:
结果忽略了 2018 年的记录,因为我们只关心从 2019-01-01 到 2019-12-31 的记录
从 2019-01-01 到 2019-01-03 所有任务成功,系统状态为 “succeeded”。
从 2019-01-04 到 2019-01-05 所有任务失败,系统状态为 “failed”。
从 2019-01-06 到 2019-01-06 所有任务成功,系统状态为 “succeeded”。

5、代码编写

网友写法

SELECT state AS period_state, 
       MIN(date) AS start_date, 
       MAX(date) AS end_date
FROM (
    SELECT *,
           row_number() over (partition by state order by date asc) AS rk1,
           row_number() over (order by date asc) AS rk2
    FROM (
        SELECT fail_date AS 'date', 'failed' AS state FROM failed
        UNION ALL
        SELECT success_date, 'succeeded' FROM succeeded
    ) t
) t2
WHERE date BETWEEN '2019-01-01' AND '2019-12-31'
GROUP BY state, rk2-rk1

代码分析

第一步,将两个表数据查询出来并标上状态,方便后面用窗口函数处理

SELECT fail_date AS 'date', 'failed' AS state FROM failed
UNION ALL
SELECT success_date, 'succeeded' FROM succeeded
datestate
2018-12-28failed
2018-12-29failed
2019-01-04failed
2019-01-05failed
2018-12-30succeeded
2018-12-31succeeded
2019-01-01succeeded
2019-01-02succeeded
2019-01-03succeeded
2019-01-06succeeded

第二步,第一个 row_number 根据状态的不同,分组后按日期正序进行排序处理,第二个 row_number 很明显是按日期正序,后面使用

SELECT *,
        row_number() over (partition by state order by date asc) AS rk1,
        row_number() over (order by date asc) AS rk2
FROM (
    SELECT fail_date AS 'date', 'failed' AS state FROM failed
    UNION ALL
    SELECT success_date, 'succeeded' FROM succeeded
) t
datestaterk1rk2
2018-12-28failed11
2018-12-29failed22
2018-12-30succeeded13
2018-12-31succeeded24
2019-01-01succeeded35
2019-01-02succeeded46
2019-01-03succeeded57
2019-01-04failed38
2019-01-05failed49
2019-01-06succeeded610

第三步,首先根据状态分组能做状态区分,第二点是关键,rk2-rk1,因为rk2是连续的,所以很清楚就能知道 rk2-rk1 如果是相等的那必定是连续的,但是可能会碰到 state 不同,rk2-rk1 相同的情况(前面先 group by state 就是首先排除这种情况)

SELECT state AS period_state, 
       MIN(date) AS start_date, 
       MAX(date) AS end_date
FROM (
    SELECT *,
           row_number() over (partition by state order by date asc) AS rk1,
           row_number() over (order by date asc) AS rk2
    FROM (
        SELECT fail_date AS 'date', 'failed' AS state FROM failed
        UNION ALL
        SELECT success_date, 'succeeded' FROM succeeded
    ) t
) t2
WHERE date BETWEEN '2019-01-01' AND '2019-12-31'
GROUP BY state, rk2-rk1
period_statestart_dateend_date
succeeded2019-01-012019-01-03
failed2019-01-042019-01-05
succeeded2019-01-062019-01-06

http://www.kler.cn/a/271965.html

相关文章:

  • 煤矿场景下拖链检测数据集VOC+YOLO格式21407张1类别
  • Go Fx 和 Java Spring 的区别
  • ⽤vector数组实现树的存储(孩⼦表示法)c++
  • Mac开启任何来源安装配置环境变量
  • 如何给自己的域名配置免费的HTTPS How to configure free HTTPS for your domain name
  • AIGC视频生成模型:ByteDance的PixelDance模型
  • Springboot 整合 Elasticsearch(五):使用RestHighLevelClient操作ES ②
  • ClickHouse中的设置的分类
  • 【LeetCode热题100】24. 两两交换链表中的节点(链表)
  • 树与二叉树(数据结构)
  • 前端学习之css伪元素选择器
  • sqlplus设置提示符
  • 【CenterFusion】模型的创建、导入、保存CenterFusion/src/lib/model/model.py
  • ApplicationListener 注册监听器来监听应用程序中发布的事件
  • 【Web开发】CSS教学(超详细,满满的干货)
  • C#八皇后算法:回溯法 vs 列优先法 vs 行优先法 vs 对角线优先法
  • 如何在WordPress网站上设置多语言展示
  • 系列学习前端之第 5 章:学习 ES6 ~ ES11
  • C语言经典面试题目(七)
  • 【Java刷题篇】串联所有单词的子串
  • Java常见问题:编辑tomcat运行环境、部署若伊系统
  • springboot使用socket和端口启动gRPC服务器的比较
  • 【计算机网络】什么是http?
  • 2.3 性能度量
  • 柔性数组(变长数组)介绍
  • 【C语言】字符函数与字符串函数以及内存函数 { 超详细攻略,一篇学会 }