DB2:SQL:SCD类型2表的粒度更改

问题描述

我有一个SCD type-2订单表,如下所示(订单日期中的交货日期,当交货日期更改时会创建历史记录)

Order_Id Order_Status Order_Create_Dt Delivery_Dt Start_Date End_Date
O1       Open         20200303        20200321    20203001   20200306
O1       Open         20200303        20200320    20200307   20200311
O1       Open         20200303        20200318    20200312   99991231

接下来,我有一个折线表,该线表也在下面的type-2行中(当数量更改或描述更改时,它具有历史记录)

Order_Id Line_Item_Id Line_Item_Desc Quantity Start_Date End_Date
O1       L1           ABC            1        20200303   20200304
O1       L1           ABC            4        20200305   99991231
O1       L2           DEF            2        20200303   99991231
O1       L3           XYZ            3        20200303   99991231

现在,作为设计更改的一部分,Delivery_Dt粒度从订单级别更改为订单项级别,并且在重新设计的订单项表历史中,需要正确捕获订单项级别更改+交货日期更改的历史记录,如下所示

Order_Id Line_Item_Id Line_Item_Desc Quantity Delivery_Dt Start_Date End_Date
O1       L1           ABC            1        20200321    20200303   20200304
O1       L1           ABC            4        20200321    20200305   20200306
O1       L2           DEF            2        20200321    20200303   20200306
O1       L3           XYZ            3        20200321    20200303   20200306  
O1       L1           ABC            4        20200320    20200307   20200311
O1       L2           DEF            2        20200320    20200307   20200311
O1       L3           XYZ            3        20200320    20200307   20200311  
O1       L1           ABC            4        20200318    20200312   99991231
O1       L2           DEF            2        20200318    20200312   99991231
O1       L3           XYZ            3        20200318    20200312   99991231 

能否仅通过使用现有Order和Line Item表的sql语句来实现?

我正在DB2数据库中尝试此操作。

解决方法

这可能是您的解决方案:

SELECT t1.Order_Id,t2.Line_Item_Id,t2.Line_Item_Desc,t2.Quantity,t1.Delivery_Dt,max(t1.Start_Date,t2.Start_Date) AS Start_Date,min(t1.End_Date,t2.End_Date) AS End_Date
  FROM t2 
 INNER JOIN T1 
    ON t2.Order_Id = t1.Order_Id
       AND t1.Start_Date <= t2.End_Date 
       AND t1.End_Date >= t2.Start_Date 
 WHERE max(t1.Start_Date,t2.Start_Date) <> min(t1.End_Date,t2.End_Date) 

关于该主题的博客很棒:乔恩·梅恩帕(Jon Maenpaa)的“ Fun with Date Ranges

您还可以通过Db2 Temporal Tables另外提供时间旅行SQL来做到这一点。