Как получить МАКСИМАЛЬНУЮ дату из таблицы, в которой используется UNION? - PullRequest
0 голосов
/ 28 декабря 2018

У меня есть 3 таблицы, которые нужно объединить, чтобы получить максимальную дату.

table_grade_A
ID_GRADE    GRADE      NOTE    SURVEYOR
       1     70.7      PASS         TOM
       3     51.2      FAIL         TOM

table_grade_B
ID_GRADE    SUB_GRADE_I   SUB_GRADE_II   TOTAL_GRADE     NOTE    SURVEYOR
       2           30.8           40.1          70.9     PASS     MARVOLO
       4           10.3           54.1          64.4     FAIL     MARVOLO                       
       5           41.7           20.9          62.6     FAIL      RIDDLE 

table_grade
ID_GRADE    STUDENT       TEST_DATE
       1       MIYA      2018-12-20   
       2      LAYLA      2018-12-21
       3       MIYA      2018-12-21
       4       MIYA      2018-12-22
       5     KARRIE      2018-12-28

Каждый студент может пройти разные тесты и разные тесты, хранящиеся в разных таблицах.Я использую UNION для заполнения значений из table_grade_a и table_grade_b и присоединяю их к table_grade

Мой текущий запрос:

SELECT tg.STUDENT, MAX(tg.TEST_DATE) AS 'TEST_DATE', temp_grade.* FROM `table_grade` tg 
INNER JOIN (
   SELECT ID_GRADE,GRADE,NOTE
   FROM table_grade_a 'tga'
   UNION ALL 
   SELECT ID_GRADE,TOTAL_GRADE AS GRADE,NOTE 
   FROM table_grade_a 'tgb'
)temp_grade ON tg.ID_GRADE = temp_grade.ID_GRADE 

WHERE tg.STUDENT = 'MIYA'

Результат запроса выше:

STUDENT    TEST_DATE    GRADE    NOTE
  MIYA    2018-12-22     70.7    PASS

Ожидаемый результат должен быть:

STUDENT    TEST_DATE    GRADE    NOTE
  MIYA    2018-12-22     64.4    FAIL

Ответы [ 2 ]

0 голосов
/ 28 декабря 2018

Для результата, соответствующего максимальной дате каждого учащегося:

MIN или MAX столбца не обязательно совпадают с другими значениями искомых строк, поэтому вам нужно сделать большечем просто рассчитать максимальную дату.В MySQL до версии 8 вы могли сделать что-то вроде этого, вычислив максимальные даты, затем используя их в качестве внутреннего соединения, чтобы ограничить строки теми, которые соответствуют максимальным значениям:

select
    temp_grade .*
from table_grade tg
inner join (
    select student, max(test_date) as test_date
    from table_grade
    group by student
    ) gd on tg.student = gd.student and tg.test_date = gd.test_date
INNER JOIN (
   SELECT ID_GRADE,GRADE,NOTE
   FROM table_grade_a 'tga'
   UNION ALL 
   SELECT ID_GRADE,TOTAL_GRADE AS GRADE,NOTE 
   FROM table_grade_a 'tgb'
   )temp_grade ON tg.ID_GRADE = temp_grade.ID_GRADE 
# WHERE tg.STUDENT = 'MIYA'

В MySQL v8 + выможно использовать row_number() over(...) вместо:

select
      temp_grade .*
from (
    select *
    , row_number() over(partition by student order by test_date DESC) as rn
    from table_grade
    )  tg
INNER JOIN (
   SELECT ID_GRADE,GRADE,NOTE
   FROM table_grade_a 'tga'
   UNION ALL 
   SELECT ID_GRADE,TOTAL_GRADE AS GRADE,NOTE 
   FROM table_grade_a 'tgb'
   )temp_grade ON tg.ID_GRADE = temp_grade.ID_GRADE 
where tg.rn = 1
# and tg.STUDENT = 'MIYA'
0 голосов
/ 28 декабря 2018

Проблема с вашим текущим подходом заключается в том, что вы выбираете максимальную дату, агрегат на уровне таблицы, а также запрашиваете все отдельные записи одновременно.Это имеет смысл.Одна правильная возможность будет использовать LIMIT с ORDER BY:

SELECT tg1.STUDENT, tg1.TEST_DATE, tg2.*
FROM table_grade tg1
INNER JOIN
(
    SELECT ID_GRADE, GRADE, NOTE
    FROM table_grade_a
    UNION ALL 
    SELECT ID_GRADE, TOTAL_GRADE, NOTE 
    FROM table_grade_b
) tg2
    ON tg1.ID_GRADE = tg2.ID_GRADE 
WHERE
    tg1.STUDENT = 'MIYA'
ORDER BY
    tg1.TEST_DATE DESC
LIMIT 1;
...