Znajdź unikalną liczbę dni

11

Chcę napisać zapytanie SQL, aby znaleźć liczbę unikalnych dni roboczych dla każdego pracownika z tabeli 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' |
*---------------------------------------*

Oczekiwany wynik:

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

Napisałem zapytanie sqlfiddle, które daje mi expectedwynik, ale dla ciekawości jest lepszy sposób na napisanie tego zapytania? Czy mogę korzystać z kalendarza lub tabeli 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

Wszelkie sugestie byłyby świetne.

gorliwy
źródło
dla wartości (1, 1, 'monday', 'wednesday'),(1, 2, 'monday', 'tuesday'),(1, 3, 'monday', 'tuesday');empid_1 pracował 3 różne dni (poniedziałek, wtorek, środa), skrzypce / zapytanie zwraca 4
lptr
1
@lptr it is (1, 1, 'monday', 'wednesday'),(1, 2, 'monday', 'tuesday'),(1, 3, 'friday', 'friday');
gorliwy
3
Twoje zapytanie nie działa. Jeśli zmieni 1 2 'monday' 'tuesday'się 1 2 'monday' 'wednesday'wynik powinien być jeszcze 4 dni ale zwraca 5
Nick

Odpowiedzi:

5

Musisz po prostu znaleźć przecięcie dni przepracowanych przez każdy emp_idz nich taskwe wszystkie dni tygodnia, a następnie policzyć poszczególne dni:

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

Wynik:

emp_id  distinct_days
1       4
2       5

Demo na SQLFiddle

Nacięcie
źródło
Nie widziałem twojej odpowiedzi pisząc moją. Teraz widzę, że komplikowałem sprawę bardziej niż to konieczne. Podoba mi się twoje rozwiązanie.
Thorsten Kettner
2
@ThorstenKettner tak - początkowo sam zacząłem rekursywną ścieżkę CTE, ale zdałem sobie sprawę, że za joinpomocą betweenas, ponieważ warunek łatwiej osiąga ten sam wynik ...
Nick
6

Jednym z możliwych sposobów uproszczenia instrukcji w pytaniu (skrzypce) jest użycie VALUESkonstruktora wartości tabeli i odpowiednich połączeń:

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

Ale jeśli chcesz policzyć różne dni, oświadczenie jest inne. Musisz znaleźć wszystkie dni od start_dayi do end_dayzakresu i policzyć poszczególne dni:

;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
Żorow
źródło
To zapytanie (jak w pierwotnym zapytaniu PO) nie działa, jeśli zmieni 1 2 'monday' 'tuesday' się 1 2 'monday' 'wednesday' wynik powinien być jeszcze 4 dni ale zwraca 5.
Nick
@Nick, przepraszam, nie rozumiem. Na podstawie wyjaśnień PO są 2 dni między mondayi wednesday. Czy coś brakuje?
Zhorov
zmień dane wejściowe zgodnie z opisem, a zapytanie zwróci 5. Jednak odpowiedź powinna nadal wynosić 4, ponieważ nadal działają tylko 4 unikalne dni.
Nick
@Nick, teraz rozumiem twój punkt widzenia. Ale jeśli zmienię wartości w skrzypcach PO, wynikiem będzie 5, nie 4. Ta odpowiedź sugeruje prostsze stwierdzenie. Dzięki.
Zhorov
Błędne jest również zapytanie OPs. Prawidłowa odpowiedź z tych danych wynosi 4, ponieważ istnieją tylko 4 unikalne dni.
Nick
2

Twoje zapytanie jest nieprawidłowe Spróbuj od poniedziałku do wtorku od środy do czwartku. Powinno to dać 4 dni, ale zapytanie zwraca 2 dni. Twoje zapytanie nawet nie wykrywa, czy dwa zakresy sąsiadują ze sobą, czy się pokrywają, czy też nie.

Jednym ze sposobów rozwiązania tego jest napisanie rekurencyjnego CTE, aby uzyskać wszystkie dni z zakresu, a następnie policzyć różne dni.

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;

Demo: http://sqlfiddle.com/#!18/4a5ac/16

(Jak widać, nie mogłem zastosować konstruktora wartości bezpośrednio jak w with weekdays (day_name, day_number) as (values ('monday', 1), ...). Nie wiem dlaczego. Czy to SQL Server czy ja? Cóż, przy dodatkowym zaznaczeniu działa :-)

Thorsten Kettner
źródło
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
Rahul Gossain
źródło
3
Tylko odpowiedzi na kod można prawie zawsze poprawić, dodając wyjaśnienie, w jaki sposób i dlaczego działają.
Jason Aller
1
Witamy w Stack Overflow! Chociaż ten kod może rozwiązać pytanie, w tym wyjaśnienie, w jaki sposób i dlaczego to rozwiązuje problem, naprawdę pomógłby poprawić jakość twojego postu i prawdopodobnie doprowadziłby do większej liczby głosów. Pamiętaj, że odpowiadasz na pytanie dla czytelników w przyszłości, nie tylko dla osoby zadającej teraz pytanie. Proszę edytować swoje odpowiedzi, aby dodać wyjaśnień i dać wskazówkę co zastosować ograniczenia i założenia. Z recenzji
podwójny sygnał dźwiękowy
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;
lptr
źródło
1
Poprosić Cię o napisanie opisu kodu, jak to działa?
Suraj Kumar