sql - 在记录水平上标记差异

标签 sql sql-server loops t-sql record

真正的脑筋急转弯就在这里。

我想标记 X1 之间的过渡。因此,从 A -> B、B-> A 和 B-> C、C-> B 的转变。

这是为了将我的数据集减少为仅那些具有转换的记录。

最终目标是计算 X1 A 最后一次发生的日期与 X1 B 第一次发生的日期之间的天数。

例如:

ID1 -> 转换 A 到 B,日期为 2018 年 1 月 20 日和 2018 年 2 月 1 日。那么答案必须是 x 之间的天数。

或者

示例 2:ID1 -> 转换 A 到 B,日期为 2018 年 2 月 16 日和 2018 年 3 月 1 日。那么答案必须是 x 之间的天数。

ID      X1      Date  
1       A       2018-01-01  
1       A       2018-01-20  
1       B       2018-02-01  
1       A       2018-02-15  
1       A       2018-02-16  
1       B       2018-03-01  
2       B       2018-01-01  
2       C       2018-03-05  
2       C       2018-03-06   
2       C       2018-03-08  
2       B       2018-03-20  

我已经有了这个循环,但是这个循环花费的时间太长(每条记录 0.3 秒):

DECLARE @Loop INT    
DECLARE @MAXROWS INT    
SET @Loop = 1    
SET @MAXROWS = (SELECT COUNT (*) FROM Y_Table)    
WHILE (@Loop <= @MAXROWS)    
BEGIN    
UPDATE Y_Table    
SET Y_Table.Test_MIN = (CASE WHEN     
(SELECT Y_Table.ID FROM Y_Table WHERE LIFE_CYCLE = @Loop) =   
(SELECT Y_Table.ID FROM Y_Table WHERE LIFE_CYCLE = @Loop + 1) AND  
(SELECT Y_Table.Date FROM Y_Table WHERE LIFE_CYCLE = @Loop) <   
(SELECT Y_Table.Date FROM Y_Table WHERE LIFE_CYCLE = @Loop + 1) AND  
(SELECT Y_Table.X1 FROM Y_Table WHERE LIFE_CYCLE = @Loop) <>   
(SELECT Y_Table.X1 FROM Y_Table WHERE LIFE_CYCLE = @Loop + 1)  
THEN '1' ELSE '0' END)  
FROM Y_Table  
WHERE   
LIFE_CYCLE = @Loop  
UPDATE Y_Table  
SET Y_Table.Test_MAX = (CASE WHEN   
(SELECT Y_Table.ID FROM Y_Table WHERE LIFE_CYCLE = @Loop) =   
(SELECT Y_Table.ID FROM Y_Table WHERE LIFE_CYCLE = @Loop + 1) AND  
(SELECT Y_Table.Date FROM Y_Table WHERE LIFE_CYCLE = @Loop) <   
(SELECT Y_Table.Date FROM Y_Table WHERE LIFE_CYCLE = @Loop + 1) AND  
(SELECT Y_Table.X1 FROM Y_Table WHERE LIFE_CYCLE = @Loop) <>   
(SELECT Y_Table.X1 FROM Y_Table WHERE LIFE_CYCLE = @Loop + 1)  
THEN '1' ELSE '0' END)  
FROM Y_Table  
WHERE   
LIFE_CYCLE = @Loop + 1  
PRINT @Loop  
SET @Loop = @Loop + 1  
END  

如何解决这个问题?

最佳答案

  • 使用 LEAD 获取下一行
  • 过滤到不同的地方(= 过渡)

类似这样的事情

DECLARE @data table (ID int, X1 char(1), myDate  date)

INSERT @data VALUES (1, 'A', '2018-01-01'),  
    (1, 'A', '2018-01-20'),  
    (1, 'B', '2018-02-01'),  
    (1, 'A', '2018-02-15'),  
    (1, 'A', '2018-02-16'),  
    (1, 'B', '2018-03-01'),  
    (2, 'B', '2018-01-01'),  
    (2, 'C', '2018-03-05'),  
    (2, 'C', '2018-03-06'),   
    (2, 'C', '2018-03-08'),  
    (2, 'B', '2018-03-20');

WITH GetNextRows AS (
    SELECT
        *,
        NextX1 = LEAD (X1) OVER (PARTITION BY ID ORDER BY myDate),
        NextDate = LEAD (myDate) OVER (PARTITION BY ID ORDER BY myDate)
    FROM
        @data
    )
SELECT 
    *,
    Transation = CONCAT(X1, ' -> ', NextX1),
    DaysGap = DATEDIFF(DAY, myDate, NextDate)
FROM
    GetNextRows
WHERE
    NextX1 <> X1;


ID  X1  myDate  NextX1  NextDate    Transation  DaysGap
1   A   2018-01-20  B   2018-02-01  A -> B      12
1   B   2018-02-01  A   2018-02-15  B -> A      14
1   A   2018-02-16  B   2018-03-01  A -> B      13
2   B   2018-01-01  C   2018-03-05  B -> C      63
2   C   2018-03-08  B   2018-03-20  C -> B      12

关于sql - 在记录水平上标记差异,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52890355/

相关文章:

java - 迭代 for 循环以获取字符串列表并存储元素值

c - 如何使用循环访问矩阵的元素

MySQL - 选择另一列的最小值后选择一列的最小值,并在特定范围内

sql-server - 将一行转换为列

sql - 如何在 SQL Server 2008 中最好地进行部分文本匹配

sql - 性能: multiple conditional WHERE clauses,还是动态查询?

sql-server - 如何在 SQL Server 中使用连接进行更新查询?

loops - 停止 WSO2 ESB 中的循环

c# - sql查询和下拉列表

mysql - SQL 查询 (MAX-MIN) 显示错误结果