在SQL中动态比较不同列和行中的日期
我有一组这样的数据 数据在SQL中动态比较不同列和行中的日期,sql,tsql,ssms,Sql,Tsql,Ssms,我有一组这样的数据 数据 ID Start_dt End_dt A 1/1/2010 12/31/2010 A 1/1/2011 12/31/2011 A 6/1/2012 12/31/2012 A 1/1/2014 12/31/2014 A 1/1/2016 10/31/2016 A 1/1/2018 12/31/2018 B 1/1/2016 2/29/2016 B 3/1/2016 10/31/2016
ID Start_dt End_dt
A 1/1/2010 12/31/2010
A 1/1/2011 12/31/2011
A 6/1/2012 12/31/2012
A 1/1/2014 12/31/2014
A 1/1/2016 10/31/2016
A 1/1/2018 12/31/2018
B 1/1/2016 2/29/2016
B 3/1/2016 10/31/2016
B 1/1/2017 7/31/2017
B 1/1/2019 12/31/9999
C 1/1/2017 12/31/2017
C 1/1/2017 12/31/2018
C 1/1/2019 12/31/9999
我需要创建一个查询,查看每个成员的行,将当前开始与前一个结束进行比较。如果差异小于一年,则将这两个记录视为一个连续注册,并返回组合的最小开始时间和最大结束时间,并对每个成员的所有行重复该操作。如果差异大于等于1年,则将其视为单独注册
期望的结果
ID Start_dt End_dt
A 1/1/2010 12/31/2012
A 1/1/2014 12/31/2014
A 1/1/2016 10/31/2016
A 1/1/2018 12/31/2018
B 1/1/2016 7/31/2017
B 1/1/2019 12/31/2019
C 1/1/2017 12/31/9999
下面是一个创建表查询:
if OBJECT_ID ('tempdb..#test1') is not null
drop table #test1
CREATE TABLE #test1 (
ID varchar(10),
Start_dt datetime,
End_dt datetime
);
INSERT INTO #test1 VALUES ('A', '1/1/2010', '12/31/2010')
,('A', '1/1/2011', '12/31/2011')
,('A', '6/1/2012', '12/31/2012')
,('A', '1/1/2014', '12/31/2014')
,('A', '1/1/2016', '10/31/2016')
,('A', '1/1/2018', '12/31/2018')
,('B', '1/1/2016', '2/29/2016')
,('B', '3/1/2016', '10/31/2016')
,('B', '1/1/2017', '7/31/2017')
,('B', '1/1/2019', '12/31/9999')
,('C', '1/1/2017', '12/31/2017')
,('C', '1/1/2017', '12/31/2018')
,('C', '1/1/2019', '12/31/2999')
我已经尝试解决这个问题好几天了,但是已经尝试了自连接、循环,但是还没有找到一个好的解决方案。有人能帮忙吗
谢谢大家! 您可以使用lag()
或累计max()
来获取上一个结束日期。然后将其与当前开始日期进行比较
当差距超过一年时,新的团队就开始了。对这些新组进行累加,开始获取分组id
剩下的是聚合:
select id, min(start_dt), max(end_dt)
from (select t1.*,
sum(case when prev_end_dt > dateadd(year, -1, start_dt) then 0 else 1 end) over
(partition by id order by start_dt) as grp
from (select t1.*,
max(end_dt) over (partition by id
order by start_dt
rows between unbounded preceding and 1 preceding
) as prev_end_dt
from test1 t1
) t1
) t1
group by id, grp
order by id, min(start_dt);
你可以试试这个查询
SELECT ID, StartDate, End_dt AS EndDate
FROM (
SELECT *
, LAG(End_dt) OVER(PARTITION BY ID ORDER BY ID, Start_dt, End_dt) AS PrevEnd
, DATEDIFF(DAY, LAG(End_dt) OVER(PARTITION BY ID ORDER BY ID, Start_dt, End_dt), Start_dt) AS DaysBreak
, (
CASE
WHEN DATEDIFF(DAY, LAG(End_dt) OVER(PARTITION BY ID ORDER BY ID, Start_dt, End_dt), Start_dt) > 365 THEN Start_dt
WHEN LAG(End_dt) OVER(PARTITION BY ID ORDER BY ID, Start_dt, End_dt) IS NULL THEN Start_dt
ELSE NULL
END
) AS StartDate
FROM #test1
) a
WHERE StartDate IS NOT NULL
哇!这非常有效。非常感谢你!你帮我省了几个小时的头发拉扯。就我的教育而言,这一条款在“无界前置和1前置之间的行”中起什么作用?谢谢!这是有帮助的,但没有产生正确的最大值(End_dt)我正在寻找。如果我将你的lag()函数与上面Gordon的聚合方法结合起来,它就会工作。您向我演示了如何使用lag()。我不能胜过你,因为我是新来的,但我感谢你的帮助。