Знайдіть унікальну кількість днів


11

Я хочу написати SQL-запит, щоб знайти таблицю кількості унікальних робочих днів для кожного працівника times.

*---------------------------------------*
|emp_id  task_id  start_day   end_day   |
*---------------------------------------*
|  1        1     'monday'  'wednesday' |
|  1        2     'monday'  'tuesday'   |
|  1        3     'friday'  'friday'    |
|  2        1     'monday'  'friday'    |
|  2        1     'tuesday' 'wednesday' |
*---------------------------------------*

Очікуваний вихід:

*-------------------*
|emp_id  no_of_days |
*-------------------*
|  1        4       |
|  2        5       |
*-------------------*

Я написав запит sqlfiddle, який дає мені expectedвихід, але для цікавості є кращий спосіб написати цей запит? Чи можу я використовувати каландр або таблицю Tally?

with days_num as  
(
  select
    *,
    case 
      when start_day = 'monday' then 1
      when start_day = 'tuesday' then 2
      when start_day = 'wednesday' then 3
      when start_day = 'thursday' then 4
      when start_day = 'friday' then 5
    end as start_day_num,

    case 
      when end_day = 'monday' then 1
      when end_day = 'tuesday' then 2
      when end_day = 'wednesday' then 3
      when end_day = 'thursday' then 4
      when end_day = 'friday' then 5
    end as end_day_num

  from times
),
day_diff as
(
  select
    emp_id,
    case
      when  
        (end_day_num - start_day_num) = 0
      then
        1
      else
        (end_day_num - start_day_num)
    end as total_diff
  from days_num  
)

select emp_id,
  sum(total_diff) as uniq_working_days
from day_diff
group by
  emp_id

Будь-які пропозиції були б чудовими.


для значень (1, 1, 'monday', 'wednesday'),(1, 2, 'monday', 'tuesday'),(1, 3, 'monday', 'tuesday');empid_1 працював 3 різні дні (понеділок, вівторок, середа), скрипка / запит повертається 4
lptr

1
@lptr це (1, 1, 'monday', 'wednesday'),(1, 2, 'monday', 'tuesday'),(1, 3, 'friday', 'friday');
завзятий

3
Ваш запит насправді не працює. Якщо ви змінилися 1 2 'monday' 'tuesday'на 1 2 'monday' 'wednesday'результат, це все-таки має бути 4 дні, але він повертається 5
Нік

Відповіді:


5

Потрібно в основному знайти перетин днів, відпрацьованих кожним emp_idна кожен taskз усіма тижнями, а потім порахувати різні дні:

with days_num as (
  SELECT *
  FROM (
    VALUES ('monday', 1), ('tuesday', 2), ('wednesday', 3), ('thursday', 4), ('friday', 5)
  ) AS d (day, day_no)
),
emp_day_nums as (
  select emp_id, d1.day_no AS start_day_no, d2.day_no AS end_day_no
  from times t
  join days_num d1 on d1.day = t.start_day
  join days_num d2 on d2.day = t.end_day
)
select emp_id, count(distinct d.day_no) AS distinct_days
from emp_day_nums e
join days_num d on d.day_no between e.start_day_no and e.end_day_no
group by emp_id

Вихід:

emp_id  distinct_days
1       4
2       5

Демонстрація на SQLFiddle


Я не бачив вашої відповіді, коли писав мою. Тепер я бачу, що я робив речі складнішими, ніж потрібно. Мені подобається ваше рішення.
Торстен Кеттнер

2
@ThorstenKettner так - я спочатку почав спускатися рекурсивного CTE шлях сам , але зрозумів , використовуючи joinз betweenяк умова досягає того ж результату легше ...
Нік

6

Один з можливих підходів для спрощення твердження у запитанні (скрипці) - це використання VALUESконструктора значень таблиці та відповідних приєднань:

SELECT 
   t.emp_id,
   SUM(CASE 
      WHEN d1.day_no = d2.day_no THEN 1
      ELSE d2.day_no - d1.day_no
   END) AS no_of_days
FROM times t
JOIN (VALUES ('monday', 1), ('tuesday', 2), ('wednesday', 3), ('thursday', 4), ('friday', 5)) d1 (day, day_no) 
   ON t.start_day = d1.day
JOIN (VALUES ('monday', 1), ('tuesday', 2), ('wednesday', 3), ('thursday', 4), ('friday', 5)) d2 (day, day_no) 
   ON t.end_day = d2.day
GROUP BY t.emp_id

Але якщо ви хочете порахувати різні дні, твердження відрізняється. Вам потрібно знайти всі дні між start_dayі end_dayдіапазоном і порахувати різні дні:

;WITH daysCTE (day, day_no) AS (
   SELECT 'monday', 1 UNION ALL
   SELECT 'tuesday', 2 UNION ALL
   SELECT 'wednesday', 3 UNION ALL
   SELECT 'thursday', 4 UNION ALL
   SELECT 'friday', 5 
)
SELECT t.emp_id, COUNT(DISTINCT d3.day_no)
FROM times t
JOIN daysCTE d1 ON t.start_day = d1.day
JOIN daysCTE d2 ON t.end_day = d2.day
JOIN daysCTE d3 ON d3.day_no BETWEEN d1.day_no AND d2.day_no
GROUP BY t.emp_id

Цей запит (як з ФОС вихідного запиту) не працює, якщо ви зміните 1 2 'monday' 'tuesday' в 1 2 'monday' 'wednesday' результаті все одно повинні бути 4 дня , але він повертає 5.
Нік

@ Nick, вибач, я не можу зрозуміти. На підставі пояснень ОП існують 2 дні між mondayі wednesday. Я щось пропускаю?
Жоров

змінити вхідні дані, як я описав, і ваш запит повернеться 5. Однак відповідь все одно має бути 4, оскільки ще працюють лише 4 унікальні дні.
Нік

@Nick, зараз я розумію твою думку. Але якщо я зміню значення в загадці ОП, результат буде 5, ні 4. Ця відповідь просто пропонує більш спрощене твердження. Дякую.
Жоров

Оперативний запит теж неправильний. Правильний відповідь з цими даними в 4, так як є тільки 4 унікальні днів.
Нік

2

Ваш запит невірний. Спробуйте з понеділка по вівторок з середи на четвер. Це має призвести через 4 дні, але ваш запит повернеться через 2 дні. Ваш запит навіть не визначає, чи два діапазони є суміжними або перекриваються чи ні.

Один із способів вирішити це - написати рекурсивний CTE, щоб отримати всі дні з діапазону, а потім підрахувати різні дні.

with weekdays (day_name, day_number) as
(
  select * from (values ('monday', 1), ('tuesday', 2), ('wednesday', 3),
                        ('thursday', 4), ('friday', 5)) as t(x,y)
)
, emp_days(emp_id, day, last_day)
as
(
  select emp_id, wds.day_number, wde.day_number
  from times t
  join weekdays wds on wds.day_name = t.start_day
  join weekdays wde on wde.day_name = t.end_day
  union all
  select emp_id, day + 1, last_day
  from emp_days
  where day < last_day
)
select emp_id, count(distinct day)
from emp_days
group by emp_id
order by emp_id;

Демо: http://sqlfiddle.com/#!18/4a5ac/16

(Як видно, я не міг застосувати конструктор значень безпосередньо як в with weekdays (day_name, day_number) as (values ('monday', 1), ...). Я не знаю, чому. Це SQL Server чи я? Ну, з додатковим вибором він працює :-)


2
with cte as 
(Select id, start_day as day
   group by id, start_day
 union 
 Select id, end_day as day
   group by id, end_day
)

select id, count(day)
from cte
group by id

3
Відповіді на код тільки майже завжди можна покращити, додавши пояснення про те, як і чому вони працюють.
Джейсон Аллер

1
Ласкаво просимо до переповнення стека! Хоча цей код може вирішити питання, включаючи пояснення, як і чому це вирішує проблему, справді допоможе покращити якість вашої публікації та, ймовірно, призведе до збільшення кількості голосів. Пам'ятайте, що ви відповідаєте на запитання читачів у майбутньому, а не лише про людину, яка зараз задає питання. Будь ласка, відредагуйте свою відповідь, щоб додати пояснення та вказати, які обмеження та припущення застосовуються. З перегляду
подвійний звуковий сигнал

1
declare @times table
(
  emp_id int,
  task_id int,
  start_day varchar(50),
  end_day varchar(50)
);

insert into @times(emp_id, task_id, start_day, end_day)
values
(1, 1, 'monday', 'wednesday'),
(1, 2, 'monday', 'tuesday'),
(1, 3, 'friday', 'friday'),
--
(2, 1, 'monday', 'friday'),
(2, 2, 'tuesday', 'wednesday'),
--
(3, 1, 'monday', 'wednesday'),
(3, 2, 'monday', 'tuesday'),
(3, 3, 'monday', 'tuesday');

--for sql 2019, APPROX_COUNT_DISTINCT() eliminates distinct sort (!!)...
-- ...with a clustered index on emp_id (to eliminate the hashed aggregation) the query cost gets 5 times cheaper ("overlooking" the increase in memory) !!??!!
/*
select t.emp_id, APPROX_COUNT_DISTINCT(v.val) as distinctweekdays
from
(
select *, .........
*/


select t.emp_id, count(distinct v.val) as distinctweekdays
from
(
select *, 
case start_day when 'monday' then 1
      when 'tuesday' then 2
      when 'wednesday' then 3
      when 'thursday' then 4
      when 'friday' then 5
    end as start_day_num,
case end_day when 'monday' then 1
      when 'tuesday' then 2
      when 'wednesday' then 3
      when 'thursday' then 4
      when 'friday' then 5
    end as end_day_num
from @times
) as t
join (values(1),(2), (3), (4), (5)) v(val) on v.val between t.start_day_num and t.end_day_num
group by t.emp_id;

1
Попросити написати опис свого коду, як він працює?
Сурай Кумар
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.