【问题标题】:Select sequential rows from ROW_Number从 ROW_Number 中选择连续行
【发布时间】:2020-12-27 05:32:52
【问题描述】:

我正在从 SQL Server 数据库中提取工资系统的数据。

我有以下疑问:

;WITH emp AS
(
    SELECT 
        [USER_ID],
        [LOG_TIME],
        CAST([LOG_TIME] AS DATE)  AS [PunchDate],
        [DEVICE_SERIAL_NO],
        [AUTHORIZE_REASON_CODE],
        [STRINGCLOCKING_TYPE0],
        ROW_NUMBER() OVER (PARTITION BY [USER_ID], CAST([LOG_TIME] AS DATE)
                           ORDER BY [USER_ID], [LOG_TIME]) AS [RowNumber]
    FROM 
        [SekureTime].dbo.USER_TIME_LOG 
    WHERE 
        ([LOG_TIME] >= DATEADD(DAY, DATEDIFF(DAY, 0, GETDATE() - 31), 0) 
         AND [DEVICE_SERIAL_NO] = '34007965' 
         AND [AUTHORIZE_REASON_CODE] = 'Access')
        OR 
        ([LOG_TIME] >= DATEADD(DAY, DATEDIFF(DAY, 0, GETDATE() - 31), 0) 
         AND [EDIT_USER] = '4' 
         AND [CLOCKING_TYPE0] = '11') 
        OR 
        ([LOG_TIME] >= DATEADD(DAY, DATEDIFF(DAY, 0, GETDATE() - 31), 0) 
         AND [EDIT_USER] = '4'
         AND [CLOCKING_TYPE0] = '8')
)
SELECT 
    t1.[USER_ID] AS [EMPID],
    t1.[PunchDate],
    t1.[DEVICE_SERIAL_NO],
    t1.[AUTHORIZE_REASON_CODE],
    t1.[STRINGCLOCKING_TYPE0],
    CONVERT(TIME(0), MIN(t1.[LOG_TIME]))  AS [punch1],
    CONVERT(TIME(0), MAX(t2.[LOG_TIME])) AS [punch2],
    SUM(ISNULL(DATEDIFF(MI, t1.[LOG_TIME], t2.[LOG_TIME]), 0)) AS [TotalMinutes]
FROM  
    emp AS t1
LEFT JOIN 
    emp AS t2 ON (t1.[USER_ID] = t2.[USER_ID]
                  AND t1.[PunchDate] = t2.[PunchDate]
                  AND t1.[RowNumber] = (t2.[RowNumber] - 1)
                  AND t2.[RowNumber] % 2 = 0)
GROUP BY 
    t1.[USER_ID],
    t1.[PunchDate],
    t1.[DEVICE_SERIAL_NO],
    t1.[AUTHORIZE_REASON_CODE],
    t1.[STRINGCLOCKING_TYPE0]

当我让员工一天只打 2 次但他们中的一些人实际上打 4 次时效果很好(午餐打卡,午餐打卡)。

我怎样才能得到那些日子的 4 拳?而不是只有第一个(MIN)和最后一个(MAX)?

预期的结果应该是这样的:

Punch Date  Emp Number  PunchDetail In      Out
9/5/2020    30919       47590       10:00 AM    7:30 PM
9/6/2020    32246       47591       10:45 AM    7:00 PM
9/6/2020    34015       47592       10:30 AM    7:00 PM
9/6/2020    34334       47593       10:15 AM    2:15 PM  <--
9/6/2020    34334       47594        2:55 AM    7:15 PM  <--
9/7/2020    34350       47595       10:30 AM    7:00 PM
9/7/2020    34792       47596       10:30 AM    7:15 PM

在哪里

【问题讨论】:

  • 您确实需要关注minimal 部分:stackoverflow.com/help/minimal-reproducible-example (将您的查询简化为与您的问题相关的部分'遇到困难,剩下的就是分散注意力,让人不想尝试)
  • 并显示一些示例数据/预期结果。
  • 感谢 @marc_s 整理您的代码,我可能会尝试一下。但是有几个问题......如果他们登录但没有退出,你想为TotalMinutes 显示什么?或者如果他们在一天中多次登录和注销?是否都必须排在一行,或者将每个输入输出对作为自己的行是否可以接受?你怎么能确定第一个事件肯定是登录,而不是通宵工作的人?
  • @MatBailie 将预期结果添加到问题中。谢谢

标签: sql sql-server row-number


【解决方案1】:

作为十个人的天真初学者,做出以下假设:

  • 第一个日志永远不会注销
  • 出拳总是成对显示
  • 如果有人登录但未退出,则忽略该日志

无需加入,只需聚合和用例语句...

;WITH emp AS
(
    SELECT 
        [USER_ID],
        [LOG_TIME],
        CAST([LOG_TIME] AS DATE)  AS [PunchDate],
        [DEVICE_SERIAL_NO],
        [AUTHORIZE_REASON_CODE],
        [STRINGCLOCKING_TYPE0],
        ROW_NUMBER() OVER (PARTITION BY [USER_ID], CAST([LOG_TIME] AS DATE)
                           ORDER BY [USER_ID], [LOG_TIME]) AS [RowNumber]
    FROM 
        [SekureTime].dbo.USER_TIME_LOG 
    WHERE 
        ([LOG_TIME] >= DATEADD(DAY, DATEDIFF(DAY, 0, GETDATE() - 31), 0) 
         AND [DEVICE_SERIAL_NO] = '34007965' 
         AND [AUTHORIZE_REASON_CODE] = 'Access')
        OR 
        ([LOG_TIME] >= DATEADD(DAY, DATEDIFF(DAY, 0, GETDATE() - 31), 0) 
         AND [EDIT_USER] = '4' 
         AND [CLOCKING_TYPE0] = '11') 
        OR 
        ([LOG_TIME] >= DATEADD(DAY, DATEDIFF(DAY, 0, GETDATE() - 31), 0) 
         AND [EDIT_USER] = '4'
         AND [CLOCKING_TYPE0] = '8')
),
    pivotted AS
(
    SELECT 
        emp.[USER_ID] AS [EMPID],
        emp.[PunchDate],
        emp.[DEVICE_SERIAL_NO],
        emp.[AUTHORIZE_REASON_CODE],
        emp.[STRINGCLOCKING_TYPE0],
        CONVERT(TIME(0), MAX(CASE WHEN emp.[RowNumber] % 2 = 1 THEN emp.[LOG_TIME] END))  AS [punch1],
        CONVERT(TIME(0), MAX(CASE WHEN emp.[RowNumber] % 2 = 0 THEN emp.[LOG_TIME] END))  AS [punch2]
    FROM  
        emp
    GROUP BY 
        emp.[USER_ID],
        emp.[PunchDate],
        emp.[DEVICE_SERIAL_NO],
        emp.[AUTHORIZE_REASON_CODE],
        emp.[STRINGCLOCKING_TYPE0],
        (emp.[RowNumber]-1) / 2
)
SELECT
    *,
    ISNULL(DATEDIFF(MI, [punch1], [punch2]), 0)   AS [TotalMinutes]
FROM
    pivotted

GROUP BY (emp.[RowNumber]-1) / 2 由于整数运算,确保所有内容都按顺序分组。

1  =>  (1-1)/2  =>  0/2  =>  0
2  =>  (2-1)/2  =>  1/2  =>  0
3  =>  (3-1)/2  =>  2/2  =>  1
4  =>  (4-1)/2  =>  3/2  =>  1
5  =>  (5-1)/2  =>  4/2  =>  2

【讨论】:

  • 我看到你在那里做了什么,但你的查询在同一行显示了 4 个拳头。有没有办法用打孔 3 和 4 代替第二行(同一日期)?
  • @VinicioGuzman - 是的,GROUP BY (emp.[RowNumber]-1) / 2 会将所有内容配对。我已经编辑了答案
猜你喜欢
  • 2016-01-27
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-01-06
  • 1970-01-01
  • 1970-01-01
  • 2010-11-26
  • 1970-01-01
相关资源
最近更新 更多