明智地按小时(24小时)进行一次客户计数

时间:2018-07-11 06:46:39

标签: sql sql-server

我有一个数据集,其中有客户ID,客户加入时间和剩余时间。我想按小时数计算每个日期的客户 这是示例数据集enter image description here

我的预期输出

在这里,我将添加我尝试过的代码片段,其中1st创建了24小时跨度,然后尝试加入并汇总函数以获取预期结果并获得当前日期,但是我需要任何日期(即动态地)

select logdate as date,timespan,count(customer_id)
(
SELECT userid,cast(joinTime as date) as logdate,customer_id
,starttime,endtime,timespan
FROM login_out_logs AS logTable 
left join 

(select '00:00:00 - 01:00:00' timespan,DATEadd(hh,0,cast(dateadd(dd,-1,getdate()))) starttime,dateadd(hh,1,cast(dateadd(dd,-1,getdate()))) endtime
union 
select '01:00:00 - 02:00:00', dateadd(hh,1,cast(dateadd(dd,-1,getdate()))),dateadd(hh,2,cast(dateadd(dd,-1,getdate())))
union 
select '02:00:00 - 03:00:00', dateadd(hh,2,cast(dateadd(dd,-1,getdate()))),dateadd(hh,3,cast(dateadd(dd,-1,getdate())))
union 
select '03:00:00 - 04:00:00', dateadd(hh,3,cast(dateadd(dd,-1,getdate()))),dateadd(hh,4,cast(dateadd(dd,-1,getdate())))
union 
select '04:00:00 - 05:00:00', dateadd(hh,4,cast(dateadd(dd,-1,getdate()))),dateadd(hh,5,cast(dateadd(dd,-1,getdate())))
union 
select '05:00:00 - 06:00:00',dateadd(hh,5,cast(dateadd(dd,-1,getdate()))),dateadd(hh,6,cast(dateadd(dd,-1,getdate())))
union 
select '06:00:00 - 07:00:00',dateadd(hh,6,cast(dateadd(dd,-1,getdate()))),dateadd(hh,7,cast(dateadd(dd,-1,getdate())))
union 
select '07:00:00 - 08:00:00',dateadd(hh,7,cast(dateadd(dd,-1,getdate()))),dateadd(hh,8,cast(dateadd(dd,-1,getdate())))
union 
select '08:00:00 - 09:00:00',dateadd(hh,8,cast(dateadd(dd,-1,getdate()))),dateadd(hh,9,cast(dateadd(dd,-1,getdate())))
union 
select '09:00:00 - 10:00:00',dateadd(hh,9,cast(dateadd(dd,-1,getdate()))),dateadd(hh,10,cast(dateadd(dd,-1,getdate())))
union 
select '10:00:00 - 11:00:00',dateadd(hh,10,cast(dateadd(dd,-1,getdate()))),dateadd(hh,11,cast(dateadd(dd,-1,getdate())))
union 
select '11:00:00 - 12:00:00',dateadd(hh,11,cast(dateadd(dd,-1,getdate()))),dateadd(hh,12,cast(dateadd(dd,-1,getdate())))
union 
select '12:00:00 - 13:00:00',dateadd(hh,12,cast(dateadd(dd,-1,getdate()))),dateadd(hh,13,cast(dateadd(dd,-1,getdate())))
union 
select '13:00:00 - 14:00:00',dateadd(hh,13,cast(dateadd(dd,-1,getdate()))),dateadd(hh,14,cast(dateadd(dd,-1,getdate())))
union 
select '14:00:00 - 15:00:00',dateadd(hh,14,cast(dateadd(dd,-1,getdate()))),dateadd(hh,15,cast(dateadd(dd,-1,getdate())))
union
select '15:00:00 - 16:00:00',dateadd(hh,15,cast(dateadd(dd,-1,getdate()))),dateadd(hh,16,cast(dateadd(dd,-1,getdate())))
union
select '16:00:00 - 17:00:00',dateadd(hh,16,cast(dateadd(dd,-1,getdate()))),dateadd(hh,17,cast(dateadd(dd,-1,getdate())))
union
select '17:00:00 - 18:00:00',dateadd(hh,17,cast(dateadd(dd,-1,getdate()))),dateadd(hh,18,cast(dateadd(dd,-1,getdate())))
union 
select '18:00:00 - 19:00:00',dateadd(hh,18,cast(dateadd(dd,-1,getdate()))),dateadd(hh,19,cast(dateadd(dd,-1,getdate())))
union
select '19:00:00 - 20:00:00',dateadd(hh,19,cast(dateadd(dd,-1,getdate()))),dateadd(hh,20,cast(dateadd(dd,-1,getdate())))
union
select '20:00:00 - 21:00:00',dateadd(hh,20,cast(dateadd(dd,-1,getdate()))),dateadd(hh,21,cast(dateadd(dd,-1,getdate())))
union
select '21:00:00 - 22:00:00',dateadd(hh,21,cast(dateadd(dd,-1,getdate()))),dateadd(hh,22,cast(dateadd(dd,-1,getdate())))
union
select '22:00:00 - 23:00:00',dateadd(hh,22,cast(dateadd(dd,-1,getdate()))),dateadd(hh,23,cast(dateadd(dd,-1,getdate())))
union
select '24:00:00 - 00:00:00',dateadd(hh,23,cast(dateadd(dd,-1,getdate()))),dateadd(hh,23,dateadd(mi,59,cast(dateadd(dd,-1,getdate())))))a
on starttime between jointime and leaveTime
or endtime between jointime and leaveTime
or jointime>=starttime and jointime<endtime

) as T
group by leaveTime,timespan



Date             Hour   customer_count
  2018-01-01      8-9    1       
  2018-01-01      9-10   1
  2018-01-01      10-11  1 
  2018-01-01      11-12  1
  2018-01-01      12-13  1
  2018-01-01      13-14  1
  2018-01-01      14-15  1
  2018-01-01      15-16  1
  2018-01-01      16-17  1
  2018-01-01      17-18  1
  2018-01-01      18-19  1
  2018-01-01      19-20  1
  2018-01-01      20-21  2
  2018-01-01      21-22  3
  2018-01-01      22-23  2
  2018-01-01      23-00  1

2 个答案:

答案 0 :(得分:1)

这是一种方法-也许这已经解决了您的问题。我设计它的目的是为了解决加入和离开之间的任何日差。但是,由于我仅使用您的示例进行了测试,因此我无法透露有关大型数据集性能的任何信息,如果涉及较大的数据集,则所有相关时间的评估可能会花费更长的时间。 无论如何,我在这里使用了递归cte来评估加入和离开之间的所有时间,之后我按日期和小时分组:

DECLARE @Cust TABLE(
  customer_id INT,
  joinTime DATETIME,
  leaveTime DATETIME
)

INSERT INTO @Cust VALUES
  (536, '2018-01-01 08:05:00', '2018-01-01 18:31:00'),
  (344, '2018-01-01 19:37:00', '2018-01-01 20:16:00'),
  (344, '2018-01-01 19:49:00', '2018-01-01 20:00:00'),
  (899, '2018-01-01 20:49:00', '2018-01-01 21:14:00'),
  (2336, '2018-01-01 21:02:00', '2018-01-01 21:03:00'),
  (335, '2018-01-01 21:03:00', '2018-01-01 23:43:00'),
  (2336, '2018-01-01 21:03:00', '2018-01-02 00:06:00'),
  (899, '2018-01-01 21:18:00', '2018-01-01 22:24:00'),
  (345, '2018-01-01 21:21:00', '2018-01-01 21:39:00'),
  (345, '2018-01-01 21:53:00', '2018-01-02 00:13:00');

;WITH cte AS(
SELECT  c.customer_id,
        c.joinTime,
        c.leaveTime,
        c.joinTime x
  FROM @Cust c
UNION ALL
SELECT  c.customer_id,
        c.joinTime,
        c.leaveTime,
        DATEADD(HOUR, 1, x) x
  FROM cte c
  WHERE DATEADD(HOUR, 1, x) <= CASE WHEN DATEPART(MINUTE, x) < DATEPART(MINUTE, c.leaveTime) THEN c.leaveTime ELSE DATEADD(HOUR, 1, c.leaveTime) END
)
SELECT CONVERT(DATE, x) AS cDate, DATEPART(HOUR, x) AS cHour, COUNT(*) AS cCount
  FROM cte
  GROUP BY CONVERT(DATE, x), DATEPART(HOUR, x)
  ORDER BY 1,2
  OPTION (MAXRECURSION 0)

答案 1 :(得分:0)

尝试一下:

Eloquent

注意:这仅适用于加入时间和休假时间相差0或1天,而不是2或更多天的情况。