Как правильно подвести итог как «время болезни» из другой таблицы - mysql - PullRequest
0 голосов
/ 19 марта 2019

В базе данных mysql я создал таблицу "Sickness":

    +--------+---------+---------+-------------+---------+-------------------------------+
    | Id_SICK|ID_WORKER| FNAME   | LNAME   | BEGIN_DATE          | END_DATE              |
    +--------+---------+---------+---------+------------+--------------------+-----------+
    | 4      |   26    | ANDREW  | WORM    |2019-03-19 07:00:00  |2019-03-19 15:00:00    |  
    +--------+---------+---------+----------------------+--------------------+-----------+  
    | 5      |   25    | ADAM    | GAX     |2019-03-21 07:00:00  |2019-03-21 15:00:00    |  
    +--------+---------+---------+----------------------+--------------------------------+  

Таблица "Workers":

+--------+---------+---------+--
|ID_WORKER |  FNAME  | LNAME   |
+----------+---------+----------
| 25       |  ADAM   |  GAX    |
+----------+---------+----------
| 26       |  ANDREW |  WORM   |
+----------+---------+----------

Таблица "Orders":

+--------+---------+---------+------------+
|ID_ORDER  |  DESC_ORDER  | NUMBER_ORDER  |
+----------+---------+--------------------+
| 20       |  TEST        |  TEST         |
+----------+---------+--------------------+

И таблица "Order_status":

+--------+---------+---------+---------+-------------+--------+----------+------------+
| Id_status|ID_WORKER| ID_ORDER| BEGIN_DATE          | END_DATE          | ORDER_DONE |
+----------+---------+---------+----------+------------+---------+--------------------+
| 47       |   25    |    20   |2019-03-18 06:50:35  |2019-03-18 15:21:32|  NO        |
+----------+---------+---------+------------+---------+-------------------+-----------+ 
| 48       |   25    |    20   |2019-03-20 06:44:12  |2019-03-20 15:11:23|  NO        |
+----------+---------+---------+------------+---------+-------------------+-----------+ 
| 50       |   25    |    20   |2019-03-22 06:50:20  |2019-03-22 12:22:33|  YES        |
+----------+---------+---------+------------+---------+-------------------+-----------+ 
| 51       |   26    |    20   |2019-03-18 06:45:11  |2019-03-18 15:14:45|  NO        |
+----------+---------+---------+------------+---------+-------------------+-----------+ 
| 52       |   26    |    20   |2019-03-20 06:50:22  |2019-03-20 15:10:32|  NO       |
+----------+---------+---------+------------+---------+-------------------+-----------+ 
| 53       |   25    |    20   |2019-03-22 06:54:11  |2019-03-22 11:23:45|  YES       |
+----------+---------+---------+------------+---------+-------------------+-----------+ 

Я хотел бы подвести итог "общего времени" работников друг друга (в таблице order_status) по заказу, в том числе с суммирующим "временем болезни" из таблицы болезни.Я правильно выбрал рабочие заказы (LNAME, FNAME) (DESC_ORDER и NUMBER_ORDER) и "ОБЩЕЕ ВРЕМЯ" для рабочих заказов друг друга.Но я не могу подвести итог болезни.Я написал команду mysql ниже:

SELECT workers.FNAME, workers.LNAME, orders.NUMBER_ORDER, orders.DESC_ORDER, SEC_TO_TIME(SUM(TIME_TO_SEC(order_status.END_DATE) - TIME_TO_SEC(order_status.BEGIN_DATE))) AS 'TOTAL TIME', SEC_TO_TIME(SUM(TIME_TO_SEC(sickness.END_DATE) - TIME_TO_SEC(sickness.BEGIN_DATE))) AS 'SICKNESS TIME' FROM order_status INNER JOIN workers ON workers.ID_WORKER = order_status.ID_WORKER INNER JOIN orders ON orders.ID_ORDER = order_status.ID_ORDER INNER JOIN sickness ON sickness.ID_WORKER = workers.ID_WORKER WHERE orders.NUMBER_ORDER LIKE 'TEST' GROUP BY workers.ID_WORKER

Тогда я получил такой результат:

+--------+---------+---------+-------+------------+------------+-------------+
|  FNAME  | LNAME   |  NUMBER_ORDER  | DESC_ORDER | TOTAL TIME | SICKNESS_TIME|
+----------+---------+---------------+------------+------------+-------------+
|  ADAM   |  GAX    | TEST           | TEST       | 22:25:38   |   24:00:00   |
+----------+---------+---------------+------------+------------+-------------+
|  ANDREW |  WORM   | TEST           | TEST       | 22:52:12   |   24:00:00   |
+----------+---------+---------------+------------+------------+-------------+

Что касается «Время болезни неверно», потому что из «Болезни» послеГруппировка ID_SICK:

+--------+---------+-----+
| Id_SICK| SICKNESS TIME |
+--------+---------+-----+
| 4      |   08:00:00    |  
+--------+---------+-----+ 
| 5      |   08:00:00    |   
+--------+---------+-----+  

У меня тоже есть обобщение "ВСЕГО ВРЕМЕНИ + ВРЕМЕНИ БОЛЕЗНИ", например

TOTAL TIME: 22:25:38 
SICKNESS TIME: 8:00:00

TOTAL + SICKNESS TIME : 22;25:38 + 8:00:00 = 30:25:38

Может кто-нибудь помочь мне, как с этим бороться?Какой запрос MySQL я должен написать?Есть идеи?Спасибо за любую помощь:)

1 Ответ

2 голосов
/ 19 марта 2019

Это на самом деле ожидается. Приведенный выше запрос объединяет 4 таблицы:

  • order_status
  • болезнь
  • рабочие
  • Заказы
Таблица

" working " имеет отношение 1: N к " order_status ".

Также « работники » имеют отношение 1: N к болезни, на самом деле это не имеет значения, даже если это 1: 1 .

Запрос создает декартово произведение между таблицами выше и дублирующиеся значения столбцов из каждой таблицы могут быть в наборе результатов запроса выше.

Удалите group by и sum, чтобы увидеть набор результатов.

например, в вашем примере для id_worker 25 у вас есть 3 строки order_status, соединенные с 1 строкой болезни (значения этой строки будут продублированы 3 раза), объединенные с 1 строкой рабочих (снова они будут продублированы 3 раза), с 1 заказами строка (такая же).

Таким образом, агрегатная функция sum объединяет повторяющиеся значения.

Это будет работать только в том случае, если результирующий набор содержит уникальные строки для всех столбцов, используемых для агрегатных функций

Чтобы решить эту проблему, используйте подзапросы для агрегирования результатов:

SELECT workers.fname, 
       workers.lname, 
       order_statusAgg.number_order,
       workers.id_worker,
       order_statusAgg.desc_order, 
       SEC_TO_TIME(SUM(order_statusAgg.stime)) AS 'TOTAL TIME', 
       SEC_TO_TIME(SUM(sicknessAgg.stime)) AS 'SICKNESS TIME' 
FROM   workers 
INNER JOIN (
SELECT sickness.id_worker, SUM((Time_to_sec(sickness.end_date) - 
                       Time_to_sec(sickness.begin_date))) AS stime
FROM sickness
GROUP BY sickness.id_worker
) sicknessAgg
               ON sicknessAgg.id_worker = workers.id_worker
       INNER JOIN (
SELECT order_status.id_worker, orders.number_order, orders.desc_order, SUM((Time_to_sec(order_status.end_date) - 
                       Time_to_sec(order_status.begin_date))) AS stime
FROM order_status
           INNER JOIN orders 
               ON orders.id_order = order_status.id_order
GROUP BY order_status.id_worker
) order_statusAgg
               ON workers.id_worker = order_statusAgg.id_worker 

WHERE  order_statusAgg.number_order LIKE 'TEST'
GROUP BY workers.id_worker

Обратите внимание, что в этом случае необходимо начинать с таблицы working , так как вы хотите агрегировать различное количество строк из таблиц order_status и sickness.

Ссылки

Как объединить три стола, чтобы получить сумму

MySQL JOIN с несколькими таблицами и SUMS

Как получить сумму двух таблиц с использованием внутреннего соединения

...