如何在 Oracle SQL 上进行查询以获取时间间隔,按特定字段分组

Posted

技术标签:

【中文标题】如何在 Oracle SQL 上进行查询以获取时间间隔,按特定字段分组【英文标题】:How to do a query on Oracle SQL to get time intervals, grouping by specific fields 【发布时间】:2020-11-30 12:34:29 【问题描述】:

我喜欢一个很好的挑战,但这个挑战让我头疼太久了。 :)

我正在尝试构建一个查询来获取日期间隔,并按一个字段对信息进行分组。

让我试着用一种简单的方式来解释它。 我们有这张桌子:

我需要得到一个士兵在每个排名上花费的时间间隔,所以我需要得到的最终结果应该是这样的:

如您所见,士兵可以随着时间的推移而被提升/降级。

关于如何构建查询来执行此操作的任何建议?

谢谢!

【问题讨论】:

请勿发布代码、数据、错误消息等的图片 - 将文本复制或输入到问题中。 How to Ask 【参考方案1】:

从 Oracle 12 开始,您可以使用MATCH_RECOGNIZE

SELECT *
FROM   table_name
MATCH_RECOGNIZE (
  PARTITION BY id
  ORDER     BY start_date, end_date
  MEASURES
    FIRST( name )       AS name,
    FIRST( ranking )    AS ranking,
    FIRST( start_date ) AS start_date,
    LAST( end_Date )    AS end_Date
  PATTERN ( same_rank+ )
  DEFINE same_rank AS FIRST( ranking ) = ranking
)

其中,对于样本数据:

CREATE TABLE table_name ( id, name, ranking, start_date, end_date ) AS
SELECT 1001, 'Jones', 'Lieutenant', DATE '2000-03-20', DATE '2002-08-15' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Lieutenant', DATE '2002-08-16', DATE '2003-03-18' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Lieutenant', DATE '2003-03-19', DATE '2004-06-01' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Lieutenant', DATE '2004-06-02', DATE '2004-10-01' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Captain',    DATE '2004-10-02', DATE '2005-04-20' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Captain',    DATE '2005-04-21', DATE '2007-02-20' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Major',      DATE '2007-02-21', DATE '2008-10-22' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Major',      DATE '2008-10-23', DATE '2010-01-26' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Captain',    DATE '2010-01-27', DATE '2013-11-25' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Captain',    DATE '2013-11-26', DATE '2014-05-11' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'Major',      DATE '2014-05-12', DATE '2016-04-22' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'General',    DATE '2016-04-23', DATE '2020-10-10' FROM DUAL UNION ALL
SELECT 1001, 'Jones', 'General',    DATE '2020-10-11', DATE '2020-11-30' FROM DUAL;

输出:

身份证 |姓名 |排名 | START_DATE |结束日期 ---: | :---- | :--------- | :----------------- | :----------------- 1001 |琼斯 |中尉 | 2000-03-20 00:00:00 | 2004-10-01 00:00:00 1001 |琼斯 |船长 | 2004-10-02 00:00:00 | 2007-02-20 00:00:00 1001 |琼斯 |专业 | 2007-02-21 00:00:00 | 2010-01-26 00:00:00 1001 |琼斯 |船长 | 2010-01-27 00:00:00 | 2014-05-11 00:00:00 1001 |琼斯 |专业 | 2014-05-12 00:00:00 | 2016-04-22 00:00:00 1001 |琼斯 |一般 | 2016-04-23 00:00:00 | 2020-11-30 00:00:00

db小提琴here

【讨论】:

惊人的解决方案!非常感谢!【参考方案2】:

这是一种间隙和孤岛问题。您想查找相同的行组,您可以使用lag() 比较ranking,然后使用累积总和来跟踪更改:

select soldier_id, soldier_name, ranking,
       min(start_date), max(end_date)
from (select t.*,
             sum(case when prev_end_date = start_date - interval '1' day then 0 else 1 end)
                 (partition by soldier_id order by start_date) as island
      from (select t.*,
                   lag(end_date) over (partition by soldier_id, ranking order by start_date) as prev_end_date
            from t
           ) t
      ) t
group by soldier_id, soldier_name, ranking, island;

注意:这假设soldier_name 对于给定士兵不会随时间变化。如果这是您需要处理的问题,请提出一个新的问题,并提供适当的样本数据和所需的结果。

【讨论】:

以上是关于如何在 Oracle SQL 上进行查询以获取时间间隔,按特定字段分组的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Oracle SQL Developer 中生成带有子查询的 INSERT 语句?

如何编写oracle SQL查询以特定顺序获取匹配和不匹配的行对(基于键列)

如何在 MySQL 上进行 SQL 区分大小写的字符串比较?

在 Oracle 中执行大型查询并返回行

Django - 如何以有效的方式获取最新的相关模型,我可以在模板上进行迭代?

查询以获取 Oracle 10g 中下一行中的 % 符号数 = 字符串长度