Roh*_*hit 4 sql sql-server group-by self-join
我有一个包含这样数据的表
+-------------+--------------+------------+----------------+
| CustomerSID | StartDateSID | EndDateSID | MarketingOptIn |
+-------------+--------------+------------+----------------+
| 12345 | 20101019 | 20131016 | Y |
| 12345 | 20131017 | 20140413 | Y |
| 12345 | 20140414 | 20140817 | N |
| 12345 | 20140818 | 20141228 | N |
| 12345 | 20141229 | 20150125 | Y |
| 12345 | 20150126 | 0 | Y |
+-------------+--------------+------------+----------------+
Run Code Online (Sandbox Code Playgroud)
我需要在这个表的顶部创建一个视图,使数据格式化为Flag的以下格式,基本上是Flag为Y或N的持续时间.(EndDateSID - 0当前是活动的,所以今天的日期)
+-------------+--------------+------------+----------------+
| CustomerSID | StartDateSID | EndDateSID | MarketingOptIn |
+-------------+--------------+------------+----------------+
| 12345 | 20101019 | 20140413 | Y |
| 12345 | 20140414 | 20141228 | N |
| 12345 | 20141229 | 20150825 | Y |
+-------------+--------------+------------+----------------+
Run Code Online (Sandbox Code Playgroud)
大多数客户的Flag只有一次更改,因此在查询工作之下:
SELECT
CH1.CustomerSID
,MIN(CH1.StartDateSID) StartDate
,MAX(ISNULL(NULLIF(CH1.EndDateSID,0),CONVERT(INT, CONVERT(VARCHAR, GETDATE(), 112)))) EndDate
,CH1.MarketingOptIn
FROM DWH.DimCustomerHistory CH1
GROUP BY CH1.CustomerSID, CH1.MarketingOptIn
ORDER BY CH1.CustomerSID, CH1.MarketingOptIn
Run Code Online (Sandbox Code Playgroud)
如何为上述客户实现预期的输出,不止一次更改标志?
编辑:根据@ GarethD的推荐,修改标题以便其他人搜索.
您可以使用以下查询:
SELECT CustomerSID,
MIN(StartDateSID) AS StartDate,
MAX(ISNULL(NULLIF(EndDateSID,0),
CONVERT(INT, CONVERT(VARCHAR, GETDATE(), 112)))) AS EndDate,
MarketingOptIn
FROM (
SELECT CustomerSID, StartDateSID, EndDateSID, MarketingOptIn,
ROW_NUMBER() OVER (ORDER BY StartDateSID) -
ROW_NUMBER() OVER (PARTITION BY CustomerSID, MarketingOptIn
ORDER BY StartDateSID) AS grp
FROM DimCustomerHistory ) AS t
GROUP BY CustomerSID, MarketingOptIn, grp
ORDER BY StartDate
Run Code Online (Sandbox Code Playgroud)
计算字段grp用于标识具有相同值的连续记录MarketingOptIn.
在外部查询中使用此字段,您可以轻松地以与原始查询类似的方式GROUP BY应用MIN和MAX聚合函数.
这是一个缺口和孤岛问题。您需要使用ROW_NUMBER()来确定您的差距,因此开始阶段将是:
SELECT CustomerSID,
StartDateSID,
EndDateSID,
MarketingOptIn,
TotalRowNum = ROW_NUMBER() OVER(PARTITION BY CustomerSID ORDER BY StartDateSID),
RowNumInGroup = ROW_NUMBER() OVER(PARTITION BY CustomerSID, MarketingOptIn ORDER BY StartDateSID),
GroupID = ROW_NUMBER() OVER(PARTITION BY CustomerSID ORDER BY StartDateSID) -
ROW_NUMBER() OVER(PARTITION BY CustomerSID, MarketingOptIn ORDER BY StartDateSID)
FROM dbo.YourTable;
Run Code Online (Sandbox Code Playgroud)
输出:
CustomerSID StartDateSID EndDateSID MarketingOptIn TotalRowNum RowNumInGroup GroupID
---------------------------------------------------------------------------------------------------
12345 20101019 20131016 Y 1 1 0
12345 20131017 20140413 Y 2 2 0
12345 20140414 20140817 N 3 1 2
12345 20140818 20141228 N 4 2 2
12345 20141229 20150125 Y 5 3 2
12345 20150126 0 Y 6 4 2
Run Code Online (Sandbox Code Playgroud)
这里的关键是,通过获取每行的行号以及组内每行的行号,您可以获得一个唯一标识符 (GroupID + MarketingOptIn),用于标识每个岛屿。那么这只是在进行聚合时按此标识符进行分组的情况:
完整的工作示例
DECLARE @T TABLE
(
CustomerSID INT,
StartDateSID INT,
EndDateSID INT,
MarketingOptIn CHAR(1)
)
INSERT @T
VALUES
(12345, 20101019, 20131016, 'Y'),
(12345, 20131017, 20140413, 'Y'),
(12345, 20140414, 20140817, 'N'),
(12345, 20140818, 20141228, 'N'),
(12345, 20141229, 20150125, 'Y'),
(12345, 20150126, 0, 'Y');
WITH CTE AS
(
SELECT CustomerSID,
StartDateSID,
EndDateSID,
MarketingOptIn,
GroupID = ROW_NUMBER() OVER(PARTITION BY CustomerSID ORDER BY StartDateSID) -
ROW_NUMBER() OVER(PARTITION BY CustomerSID, MarketingOptIn ORDER BY StartDateSID)
FROM @T
)
SELECT CustomerSID,
StartDateSID = MIN(StartDateSID),
EndDateSID = CASE WHEN MIN(EndDateSID) = 0 THEN CONVERT(INT, CONVERT(VARCHAR(8), GETDATE(), 112)) ELSE MAX(EndDateSID) END,
MarketingOptIn
FROM CTE
GROUP BY CustomerSID, MarketingOptIn, GroupID
ORDER BY CustomerSID, StartDateSID;
Run Code Online (Sandbox Code Playgroud)