T-SQL - 两个日期时间之间每小时的分钟数

phi*_*con 3 t-sql sql-server datetime date

我必须遵循以下数据:

 | tid | startdate        | enddate                |
 | 1   | 2016-12-26 12:30 | 2016-12-26 15:30       |
 | 2   | 2016-12-26 13:15 | 2016-12-26 15:15       |
Run Code Online (Sandbox Code Playgroud)

我想创建一个带有小时数的结果,然后是日期时间在该小时内的分钟数.

示例结果:

 | tid | hour | minutes_in |
 | 1   | 12   | 30         |
 | 1   | 13   | 60         |
 | 1   | 14   | 60         |
 | 1   | 15   | 30         |
 | 2   | 13   | 45         |
 | 2   | 14   | 60         |
 | 2   | 15   | 15         |
Run Code Online (Sandbox Code Playgroud)

有什么建议?

Gar*_*thD 5

首先你需要一个数字表来获得从0到23的小时数,这可以通过表值构造函数轻松地创建:

SELECT N
FROM (VALUES 
        (0),(1),(2),(3),(4),(5),(6),(7),(8),(9),(10),(11),(12),
        (13),(14),(15),(16),(17),(18),(19),(20),(21),(22),(23)
    ) n (N);
Run Code Online (Sandbox Code Playgroud)

然后,您可以将其与原始数据相连,以将行拆分为所需的数字.然后你只需要一个case表达式来应用正确的逻辑来计算分钟:

WITH Numbers (Number) AS
(   SELECT N
    FROM (VALUES 
            (0),(1),(2),(3),(4),(5),(6),(7),(8),(9),(10),(11),(12),
            (13),(14),(15),(16),(17),(18),(19),(20),(21),(22),(23)
        ) n (N)
), SampleData (tid, StartDate, EndDate) AS
(   SELECT  tid, CONVERT(DATETIME2, StartDate), CONVERT(DATETIME2, EndDate)
    FROM (VALUES 
            (1, '2016-12-26 12:30', '2016-12-26 15:30'),
            (2, '2016-12-26 13:15', '2016-12-26 15:15')
        ) d (tid, StartDate, EndDate)
)
SELECT  d.tid,
        [Hour] = n.Number,
        Minutes_in = CASE  
                        -- SPECIAL CASE: START HOUR = END HOUR
                        WHEN DATEPART(HOUR, d.StartDate) = DATEPART(HOUR, d.EndDate) 
                            THEN DATEDIFF(MINUTE, d.StartDate, d.EndDate)

                        -- FULL HOURS IN BETWEEN START AND END
                        WHEN n.Number > DATEPART(HOUR, d.StartDate) 
                            AND n.Number < DATEPART(HOUR, d.EndDate) THEN 60

                        -- START HOUR
                        WHEN n.Number = DATEPART(HOUR, d.StartDate) 
                            THEN 60 - DATEPART(MINUTE, d.StartDate)

                        -- END HOUR
                        WHEN n.Number = DATEPART(HOUR, d.EndDate) 
                            THEN DATEPART(MINUTE, d.EndDate)
                    END
FROM    SampleData d
        INNER JOIN Numbers n 
            ON n.Number >= DATEPART(HOUR, d.StartDate)
            AND n.Number <= DATEPART(HOUR, d.EndDate);
Run Code Online (Sandbox Code Playgroud)

附录

如果您需要跨越几天,那么您可以稍微改变逻辑,生成一组更大的数字以覆盖更多的小时差异,然后加入小时差异从开始日期时间到结束日期时间:

SELECT  *
FROM    SampleData d
        INNER JOIN Numbers n 
            ON n.Number <= DATEDIFF(HOUR, d.StartDate, d.EndDate)
Run Code Online (Sandbox Code Playgroud)

这意味着范围跨越几天,然后没有问题,时间只是继续递增.例如

WITH Numbers (Number) AS
(   SELECT ROW_NUMBER() OVER(ORDER BY N1.N) - 1
    FROM (VALUES (1),(1),(1),(1),(1),(1),(1),(1),(1),(1)) N1(N)
    CROSS JOIN (VALUES (1),(1),(1),(1),(1),(1),(1),(1),(1),(1)) N2 (N)
    CROSS JOIN (VALUES (1),(1),(1),(1),(1),(1),(1),(1),(1),(1)) N3 (N)
    CROSS JOIN (VALUES (1),(1),(1),(1),(1),(1),(1),(1),(1),(1)) N4 (N)
), SampleData (tid, StartDate, EndDate) AS
(   SELECT  tid, CONVERT(DATETIME2, StartDate), CONVERT(DATETIME2, EndDate)
    FROM (VALUES 
            (1, '2016-12-26 12:30', '2016-12-26 15:30'),
            (2, '2016-12-26 13:15', '2016-12-26 15:15'),
            (3, '2016-12-26 13:15', '2016-12-27 15:15')
        ) d (tid, StartDate, EndDate)
)
SELECT  d.tid,
        [Date] = CONVERT(DATE, d.StartDate),
        [Hour] = CONVERT(TIME(0), DATEADD(HOUR, DATEPART(HOUR, d.StartDate) + n.Number, 0)),
        Minutes_in = CASE  
                        -- SPECIAL CASE: START HOUR = END HOUR
                        WHEN DATEPART(HOUR, d.StartDate) = DATEPART(HOUR, d.EndDate)
                            AND DATEDIFF(DAY, d.StartDate, d.EndDate) = 0
                            THEN DATEDIFF(MINUTE, d.StartDate, d.EndDate)

                        -- START HOUR
                        WHEN n.Number = 0
                            THEN 60 - DATEPART(MINUTE, d.StartDate)

                        -- END HOUR
                        WHEN n.Number = DATEDIFF(HOUR, d.StartDate, d.EndDate) 
                            THEN DATEPART(MINUTE, d.EndDate)

                        -- FULL HOURS IN BETWEEN START AND END
                        ELSE 60

                    END
FROM    SampleData d
        INNER JOIN Numbers n 
            ON n.Number <= DATEDIFF(HOUR, d.StartDate, d.EndDate)
ORDER BY d.tid, n.Number;
Run Code Online (Sandbox Code Playgroud)