从SQL表中查找不连续性

Anl*_*nlo 8 sql t-sql sql-server sql-server-2008

对我的问题可能有一个非常简单的解决方案,但我正在为它制定一个好的搜索短语.我有一个包含时间戳和计数的表:

2013-08-15 14:43:58.447    5
2013-08-15 14:44:58.307    12
2013-08-15 14:45:58.383    14
2013-08-15 14:46:58.180    0
2013-08-15 14:47:58.210    4
2013-08-15 14:48:58.287    6
2013-08-15 14:49:58.550    12
2013-08-15 14:50:58.440    2
2013-08-15 14:51:58.390    5
Run Code Online (Sandbox Code Playgroud)

如您所见,计数增加,然后偶尔清空.搜索count = 0的行很容易,但有时计数会在记录零计数之前增加.在14:49,计数为12,然后重置为0并在14:50的下一个日志之前递增到2.

我需要列出计数小于之前的计数的时间戳:

2013-08-15 14:46:58.180    0
2013-08-15 14:50:58.440    2
Run Code Online (Sandbox Code Playgroud)

我开始在表本身上进行连接,比较两行,但SQL很快就变得非常混乱了.

pod*_*ska 10

这样做是基于ts(datetime)列创建行号,然后更容易加入到上一个条目.然后,它会比较时间和计数以查找异常.

;with cte as 
(
    select * ,
      ROW_NUMBER() over (order by ts) rn    
    from yourtable
)
    select c1.* from cte c1
        inner join cte c2
            on c1.rn=c2.rn+1
            and c1.c < c2.c
Run Code Online (Sandbox Code Playgroud)


val*_*lex 5

在这种情况下,您也可以使用LEAD()函数:

with CTE as
(
select t.*, LEAD(ct) OVER (ORDER BY dt DESC) as LEAD_CT from t
)  
select dt,ct from CTE where LEAD_CT>CT
Run Code Online (Sandbox Code Playgroud)

SQLFiddle演示

UPD:LEAD()可从SQLServer 2012版本获得.在2008年,您可以使用子查询替换它:

select *
      FROM T as T1
      where (SELECT TOP 1 ct FROM T 
                             WHERE T.dt<T1.DT
                             ORDER BY dt DESC) >CT
Run Code Online (Sandbox Code Playgroud)

SQLFiddle演示

  • 不是在SQL 2008中你不能. (2认同)
  • 但是,在MSSQL2012中有一个很好的功能!无论如何,谢谢你们.;-) (2认同)