微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

使用日期范围的可能的递归CTE查询

如何解决使用日期范围的可能的递归CTE查询

在这里不需要“真实的”递归CTE。不过,您可以使用一个作为月份参考:

;WITH Months
as
(
    SELECT DATEADD(day, -DATEPART(day, GETDATE())+1, GETDATE()) as 'MonthStart'
    UNION ALL
    SELECT DATEADD(month, -1, MonthStart) as 'MonthStart'
    FROM Months
)

然后,你可以JOINSELECT TOP 13 * FROM Months你上面的查询

我不会尝试解析所有CASE语句,但是从本质上讲,您可以GROUP BY在日期和MonthStart字段上使用a ,例如:

GROUP BY Datepart(year, monthstart), Datepart(month, monthstart)

并按月汇总。将所有选项(活动,失效等)列为一列并使用来计算每个选项可能是最容易的SUM(CASE WHEN ... THEN 1 ELSE 0 END),因为使用会更容易GROUP BY

解决方法

不知道如何在这个标题上加上标题!

我有以下数据:

IF OBJECT_ID ('tempdb..#data') IS NOT NULL DROP TABLE #data
CREATE TABLE #data
(
id UNIQUEIDENTIFIER,reference NVARCHAR(30),start_date DATETIME,end_date DATETIME,lapse_date DATETIME,value_received DECIMAL(18,3)
)

INSERT INTO #data VALUES ('BE91B9C1-C02F-46F7-9B63-4D0B25D9BA2F','168780','2006-05-01 00:00:00.000',NULL,'2011-09-27 00:00:00.000',537.42)
INSERT INTO #data VALUES ('B538F123-C839-447A-B300-5D16EACF4560','320858','2011-08-08 00:00:00.000',0)
INSERT INTO #data VALUES ('1922465D-2A55-434D-BAAA-8E15D681CF12','306597','2011-04-08 00:00:00.000','2011-06-22 13:14:40.083','2011-08-07 00:00:00.000',12)
INSERT INTO #data VALUES ('7DF8FBCC-B490-4892-BDC5-8FD2D73B0323','321461','2011-07-01 00:00:00.000','2011-09-25 00:00:00.000',8.44)
INSERT INTO #data VALUES ('1EC2E754-F325-4313-BDFC-9010E255F6FE','74215','2000-10-31 00:00:00.000','2011-08-30 00:00:00.000',258)
INSERT INTO #data VALUES ('9E59B09C-0198-48AC-8EEC-A0D76CEA9385','169194','2008-06-25 00:00:00.000',1766.4)
INSERT INTO #data VALUES ('97CF6C0F-324A-49A6-B9D8-AC848A1F821A','288039','2010-09-01 00:00:00.000','2011-07-29 00:00:00.000','2011-08-21 00:00:00.000',55)
INSERT INTO #data VALUES ('97CF6C0F-324A-49A6-B9D8-AC848A1F821A','324423','2011-08-01 00:00:00.000',5)
INSERT INTO #data VALUES ('D5E5197A-E8E1-468C-9991-C8712224C2BF','323395','2011-08-25 00:00:00.000',0)
INSERT INTO #data VALUES ('0EC4976C-16B9-4C99-BD07-D0CBDF014D32','323741',0)

我希望能够根据以下标准将所有引用分为“活动”,“失效”或“新”类别:

  • Active 起始日期小于参考月份的最后日期,失效日期在上个月的最后一天之后,并且value_received> 0;

  • New 开始日期在参考月份之内;

  • Lapsed 失效日期在参考月份之内。

然后将这些定义应用到连续13个月的每个引用中(因此从Now追溯到2010年7月),以便每个月我可以看到每个组中有多少引用。

我可以使用以下内容来定义当前月份:

select 
id,reference,start_date,end_date,lapse_date,value_received,CASE   WHEN start_date < DATEADD(month,DATEPART(Month,GETDATE()) + 1,DATEADD(year,DATEPART(year,GETDATE())-1900,0)) --next month start date
        AND lapse_date > DATEADD(ms,-3,DATEADD(mm,DATEDIFF(mm,GETDATE())+1,0)) --last day of current month
        AND value_received > 0
        THEN 'Active'
        WHEN lapse_date < DATEADD(month,0)) --next month start
            AND lapse_date > DATEADD(ms,GETDATE()),0)) --last day of prior month
        THEN 'lapse'
        WHEN start_date < DATEADD(month,0)) --next month start date
        AND start_date > DATEADD(ms,0)) --last day of prior month
        THEN 'New'
        ELSE 'Not applicable'
 END AS [type]
from #data

但是我看不到一个很好的/有效的方法(除了重复此查询13次并合并结果,我知道这很糟糕)

使用当前月份作为锚点并使用递归(如果这样的话,一些指针将是最好的选择)吗?

一如既往的帮助:)

编辑为包含实际解决方案

如果任何人都感兴趣,这是我使用的最后一个查询:

;WITH Months as 
(
SELECT DATEADD(ms,0)) as month_end,0 AS level
UNION ALL     
SELECT DATEADD(month,-1,month_end)as month_end,level + 1 FROM Months
WHERE level < 13 
) 
SELECT 
DATENAME(Month,month_end) + ' ' + DATENAME(YEAR,month_end) as date,SUM(CASE WHEN start_date <= month_end
        AND Month(start_date) <> MONTH(Month_end)
        AND lapse_date > Month_end 
 THEN 1 ELSE 0 END) AS Active,SUM(CASE WHEN start_date <= Month_end 
        AND DATENAME(MONTH,start_date) + ' ' + DATENAME(YEAR,start_date) = 
        DATENAME(MONTH,month_end)
THEN 1 ELSE 0 END) AS New,SUM(CASE WHEN lapse_date <= Month_end 
        AND Month(lapse_date) = MONTH(Month_end)
THEN 1 ELSE 0 END) AS lapse
FROM #data
CROSS JOIN Months
WHERE id IS NOT NULL
AND start_date IS NOT NULL
GROUP BY DATENAME(Month,month_end)  + ' ' + DATENAME(YEAR,month_end) 
ORDER by MAX(level) ASC

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。