10

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

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

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

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

2013-08-15 14:46:58.180    0
2013-08-15 14:50:58.440    2

我开始对表本身进行连接,以比较两行,但 SQL 很快就变得非常混乱。

4

3 回答 3

11

它的作用是根据 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
于 2013-08-15T13:05:40.363 回答
6

同样在这种情况下,您可以使用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

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

SQLFiddle 演示

于 2013-08-15T13:13:30.030 回答
1

使用ROW_NUMBER()功能,您可以在订单上分配号码并将它们用于加入:

WITH CTE_RN AS
(
    SELECT *, ROW_NUMBER() OVER (ORDER BY [TimeStamp]) RN
    FROM Table1
)
SELECT r2.* 
FROM CTE_RN r1
    INNER JOIN CTE_RN r2 ON r1.RN +1 = r2.RN
WHERE r1.count > r2.count

SQLFiddle 演示

于 2013-08-15T13:09:58.970 回答