使用SQL检测连续的日期范围

Sha*_*mim 28 sql sql-server sql-server-2008 gaps-and-islands

我想填写需要开始和结束日期信息的日历对象.我有一列包含一系列日期.有些日期是连续的(有一天差异),有些则不是.

InfoDate  

2013-12-04  consecutive date [StartDate]
2013-12-05  consecutive date
2013-12-06  consecutive date [EndDate]

2013-12-09                   [startDate]
2013-12-10                   [EndDate]

2014-01-01                   [startDate]
2014-01-02 
2014-01-03                   [EndDate]

2014-01-06                   [startDate]
2014-01-07                   [EndDate]

2014-01-29                   [startDate]
2014-01-30 
2014-01-31                   [EndDate]

2014-02-03                   [startDate]
2014-02-04                   [EndDate]
Run Code Online (Sandbox Code Playgroud)

我想选择每个连续日期范围的开始和结束日期(块中的第一个和最后一个).

StartDate     EndDate

2013-12-04    2013-12-06
2013-12-09    2013-12-10
2014-01-01    2014-01-03
2014-01-06    2014-01-07
2014-01-29    2014-01-31
2014-02-03    2014-02-04
Run Code Online (Sandbox Code Playgroud)

我想只使用SQL来解决问题.

Ano*_*non 48

不需要连接或递归CTE.标准的gap-and-island解决方案是分组(值减去row_number),因为它在连续序列中是不变的.开始日期和结束日期只是组的MIN()和MAX().

WITH t AS (
  SELECT InfoDate d,ROW_NUMBER() OVER(ORDER BY InfoDate) i
  FROM @d
  GROUP BY InfoDate
)
SELECT MIN(d),MAX(d)
FROM t
GROUP BY DATEDIFF(day,i,d)
Run Code Online (Sandbox Code Playgroud)


sat*_*esh 6

干得好..

;WITH CTEDATES
AS
(
    SELECT ROW_NUMBER() OVER (ORDER BY Infodate asc ) AS ROWNUMBER,infodate FROM YourTableName  

),
 CTEDATES1
AS
(
   SELECT ROWNUMBER, infodate, 1 as groupid FROM CTEDATES WHERE ROWNUMBER=1
   UNION ALL
   SELECT a.ROWNUMBER, a.infodate,case datediff(d, b.infodate,a.infodate) when 1 then b.groupid else b.groupid+1 end as gap FROM CTEDATES A INNER JOIN CTEDATES1 B ON A.ROWNUMBER-1 = B.ROWNUMBER
)

select min(mydate) as startdate, max(infodate) as enddate from CTEDATES1 group by groupid
Run Code Online (Sandbox Code Playgroud)

  • 您应该使用“OVER (ORDER BY Infodate)”而不是“OVER (ORDER BY (SELECT 1))”。另外,将“min(mydate)”更改为“min(infodate)”。除此之外,这是一个很好的答案 (2认同)