Encontre um número único de dias

11

Desejo escrever uma consulta SQL para encontrar o número de dias úteis exclusivos para cada funcionário da tabela 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' |
*---------------------------------------*

Saída esperada:

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

Eu escrevi a consulta sqlfiddle que está me dando a expectedsaída, mas por curiosidade, existe uma maneira melhor de escrever esta consulta? Posso usar a mesa do calendário ou da tabela?

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

Qualquer sugestão seria ótima.

zeloso
fonte
para valores (1, 1, 'monday', 'wednesday'),(1, 2, 'monday', 'tuesday'),(1, 3, 'monday', 'tuesday');empid_1 trabalhou 3 dias distintos (segunda-feira, terça-feira, quarta-feira), o violino / consulta retorna 4
lptr
11
@lptr it is (1, 1, 'monday', 'wednesday'),(1, 2, 'monday', 'tuesday'),(1, 3, 'friday', 'friday');
zealous
3
Sua consulta não funciona realmente. Se você mudar 1 2 'monday' 'tuesday'para 1 2 'monday' 'wednesday'o resultado ainda deve ser de 4 dias, mas retorna 5
Nick

Respostas:

5

Você precisa basicamente encontrar a interseção dos dias trabalhados por cada emp_idum taskcom todos os dias da semana e depois contar os dias distintos:

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

Resultado:

emp_id  distinct_days
1       4
2       5

Demonstração no SQLFiddle

usuario
fonte
Não vi sua resposta ao escrever a minha. Agora vejo que estava tornando as coisas mais complicadas do que o necessário. Eu gosto da sua solução.
Thorsten Kettner 27/03
2
@ThorstenKettner yeah - Eu comecei o caminho CTE recursivo por conta própria, mas percebi usando a joincom betweena condição de obter o mesmo resultado mais facilmente ...
Nick
6

Uma abordagem possível para simplificar a declaração na pergunta (violino) é usar o VALUESconstrutor de valor de tabela e as junções apropriadas:

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

Mas se você deseja contar os dias distintos , a declaração é diferente. Você precisa encontrar todos os dias entre a start_daye end_daygama e conta as distintas dias:

;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
Zhorov
fonte
Essa consulta (como na consulta original dos OPs) não funciona, se você alterar 1 2 'monday' 'tuesday' para 1 2 'monday' 'wednesday' o resultado ainda deve ser de 4 dias, mas retorna 5.
Nick
@ Nick, desculpe, eu não consigo entender. Com base nas explicações dos POs, existem 2 dias entre mondaye wednesday. Estou esquecendo de algo?
Zhorov 27/03
altere os dados de entrada conforme descrito, e sua consulta retornará 5. No entanto, a resposta ainda deve ser 4, pois ainda existem apenas 4 dias úteis.
Nick
@ Nick, agora eu entendo o seu ponto. Mas se eu alterar os valores no OP, o resultado será 5, não 4. Esta resposta apenas sugere uma declaração mais simples. Obrigado.
Zhorov 27/03
A consulta do OP também está errada. A resposta correta com esses dados é 4, pois existem apenas 4 dias únicos.
Nick
2

Sua consulta não está correta. Tente de segunda a terça-feira com quarta a quinta-feira. Isso deve resultar em 4 dias, mas sua consulta retorna 2 dias. Sua consulta nem detecta se dois intervalos são adjacentes ou sobrepostos ou nenhum.

Uma maneira de resolver isso é escrever uma CTE recursiva para obter todos os dias de um intervalo e depois contar dias distintos.

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;

Demonstração: http://sqlfiddle.com/#!18/4a5ac/16

(Como pode ser visto, não pude aplicar o construtor de valores diretamente como em with weekdays (day_name, day_number) as (values ('monday', 1), ...). Não sei por que. Esse é o SQL Server ou eu? Bem, com a seleção adicional funciona :-)

Thorsten Kettner
fonte
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
fonte
3
As respostas somente de código quase sempre podem ser aprimoradas com a adição de algumas explicações de como e por que elas funcionam.
Jason Aller
11
Bem-vindo ao Stack Overflow! Embora esse código possa resolver a questão, incluindo uma explicação de como e por que isso resolve o problema realmente ajudaria a melhorar a qualidade da sua postagem e provavelmente resultará em mais votos positivos. Lembre-se de que você está respondendo à pergunta dos leitores no futuro, não apenas à pessoa que está perguntando agora. Por favor edite sua resposta para adicionar explicações e dar uma indicação do que limitações e premissas se aplicam. Da avaliação
double-beep
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
fonte
11
Solicitar que você escreva uma descrição do seu código como ele funciona?
Suraj Kumar