sql - 查找序列中未缺失的列中的上一个数字

标签 sql oracle datetime window-functions gaps-and-islands

我有一个表,其中有一列应包含完整序列中的数字,为简单起见,我们将说 101 到 110。但是,该表依赖于手动输入的不可靠信息,因此会丢失序列中的数字。同一张表中还有一个日期列我需要引用,稍后会详细介绍。我面临的挑战是找到所有丢失的序列号以及输入的前一个序列号、输入日期以及下一个序列号及其输入日期。查找丢失的序列号很简单,它正在获取我正在努力处理的相关上一个和下一个记录。所以如果我的数据看起来像这样;

table, th, td {
  border: 1px solid black;
  border-collapse: collapse;
}
<html>

<body>
  <table>
    <tr>
      <th>Seq No</th>
      <th>Date Input</th>
    </tr>
    <tr>
      <td>101</td>
      <td>01-JAN-20</td>
    </tr>
    <tr>
      <td>102</td>
      <td>05-JAN-20</td>
    </tr>
    <tr>
      <td>104</td>
      <td>07-JAN-20</td>
    </tr>
    <tr>
      <td>105</td>
      <td>08-JAN-20</td>
    </tr>
    <tr>
      <td>106</td>
      <td>09-JAN-20</td>
    </tr>
    <tr>
      <td>108</td>
      <td>10-JAN-20</td>
    </tr>
    <tr>
      <td>109</td>
      <td>11-JAN-20</td>
    </tr>
    <tr>
      <td>110</td>
      <td>12-JAN-20</td>
    </tr>
  </table>
</body>

</html>

我的结果集看起来像这样;

table, th, td {
  border: 1px solid black;
  border-collapse: collapse;
}
<html>
<body>
  <table>
    <tr>
      <th>Missing Seq No</th>
      <th>Previous Date</th>
      <th>Next Date</th>
      <th>Notes</th>
    </tr>
    <tr>
      <td>103</td>
      <td>05-JAN-20</td>
      <td>07-JAN-20</td>
      <td>Dates from found seq nos 102 and 104</td>
    </tr>
    <tr>
      <td>107</td>
      <td>09-JAN-20</td>
      <td>10-JAN-20</td>
      <td>Dates from found seq nos 106 and 108</td>
    </tr>
  </table>
</body>
</html>

但是没有注释栏,这只是为了清晰起见。

我可以得到答案,但是 sql 太大而且笨拙,几乎毫无用处。谢谢。

最佳答案

如果您希望每个间隙一行,那么您可以使用窗口函数:

select 
    lag_seq_no last_sequence_number,
    lag_date_input last_date_input,
    seq_no next_sequence_number,
    date_input next_date_input
from (
    select 
        t.*,
        lag(seq_no) over(order by date_input) lag_seq_no,
        lag(date_input) over(order by date_input) lag_date_input
    from mytable t
) t
where seq_no > lag_seq_no + 1

另一方面,如果您有连续的缺失数字并且您想要每个数字一行,那么您需要某种递归:

with 
    data(seq_no, date_input, lag_seq_no, lag_date_input) as (
        select 
            t.*,
            lag(seq_no) over(order by date_input) lag_seq_no,
           lag(date_input) over(order by date_input) lag_date_input
        from mytable t
    ),
    cte (seq_no, date_input, lag_seq_no, lag_date_input) as (
        select seq_no, date_input, lag_seq_no + 1, lag_date_input
        from data
        where seq_no > lag_seq_no + 1
        union all
        select seq_no, date_input, lag_seq_no + 1, lag_date_input
        from cte
        where seq_no > lag_seq_no + 1
) 
select 
    lag_seq_no missing_seq_no,
    lag_date_input last_date_input,
    date_input next_date_input
from cte
    

关于sql - 查找序列中未缺失的列中的上一个数字,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63470179/

相关文章:

mysql - 如何在 MySQL Workbench 中设置级联删除?

oracle - 将 Oracle PL/SQL 转换为 Postgresql PL/pgSQL 的工具

MySQL-日期-时间戳

database - 字段的数据类型

sql - Hive 游标实现替代方案

sql - SQL Server SELECT 查询中括号的作用?

MySQL连接查询混淆

Oracle Last_ddl_time 格式

sql - 当args传递时,oracle查询返回0条记录

php - WPDB 查询 - 根据 _meta_key 返回 meta_value