在SQL Server的列表中标识连续数字组

时间:2018-02-02 09:46:18

标签: sql-server-2012

示例数据

Column1
-------
   1
   4
  10
  11
  12
  18
  25
  27
  28
  29
  33
  36

预期输出

Continuous Numbers
------------------
     10
     11
     12
     27
     28
     29

count(coutinous numbers)
------------------------
       2 

2 个答案:

答案 0 :(得分:0)

虽然这个问题非常糟糕(下次请阅读How to ask a good SQL questionHow to create a MCVE)我会试着指出你的方式:

使用SQL-Server 2012,您可以使用LAG()LEAD()

DECLARE @tbl TABLE(Column1 INT)
INSERT INTO @tbl VALUES(1),(4),(10),(11),(12),(18),(25),(27),(28),(29),(33),(36);

WITH Neighbours AS
(
    SELECT Column1 AS CurrentValue
          ,CASE WHEN Column1+1<>ISNULL(LEAD(Column1) OVER(ORDER BY Column1),10000) 
                 AND Column1-1<>ISNULL(LAG(Column1) OVER(ORDER BY Column1),-1) THEN 'gap' ELSE '' END AS MarkGaps
    FROM @tbl AS t
)
SELECT CurrentValue
      ,CASE WHEN CurrentValue-1<>ISNULL(LAG(CurrentValue) OVER(ORDER BY CurrentValue),10000) THEN 'grpStart' ELSE '' END
FROM Neighbours
WHERE MarkGaps = '';

结果

CurrentValue    
10  grpStart
11  
12  
27  grpStart
28  
29  

为了计算你的数量,你可以算上“grpStart”

答案 1 :(得分:0)

您需要使用潜在客户和滞后函数: 请看这个链接:SQL Server 2012 Functions - Lead and Lag

-- Declare Table to test : 
       DECLARE @Temp TABLE (Number INT) 
        INSERT INTO @Temp VALUES ( 1 ) , ( 4 ) , ( 10 ) ,( 11 ) ,( 12 ) ,
                                     ( 18 ) ,( 25 ) ,( 27 ) ,( 28 ) ,( 29 ) ,
                                     ( 33 ) ,( 36 );

---------------------------------------------------------------------------------
-- To return the count 

SELECT COUNT(1) CoutinousNumberCount
FROM   (   SELECT Number ,
                  LAG(Temp.Number, 1) OVER ( ORDER BY Temp.Number ) AS PreviousNumber ,
                  LEAD(Temp.Number, 1) OVER ( ORDER BY Temp.Number ) AS NextNumber ,
                  CASE WHEN LAG(Temp.Number, 1) 
                            OVER ( ORDER BY Temp.Number )  = Number - 1 THEN
                           0
                       ELSE 1
                  END AS Separator
           FROM   @Temp AS Temp 
       ) AS result
WHERE  (result.PreviousNumber = result.Number - 1
       OR result.NextNumber = result.Number + 1)
       AND Separator = 1;

-----------------------------------------------------------------------------------------
-- And to return the list:
SELECT Number
FROM   (   SELECT Number ,
                  LAG(Temp.Number, 1) OVER ( ORDER BY Temp.Number ) AS PreviousNumber ,
                  LEAD(Temp.Number, 1) OVER ( ORDER BY Temp.Number ) AS NextNumber 
           FROM   @Temp AS Temp
       ) AS result
WHERE  result.PreviousNumber = result.Number - 1
       OR result.NextNumber = result.Number + 1;