Запрос обновления MySQL с левым соединением и группой
Я пытаюсь создать запрос на обновление и сделать небольшой прогресс в получении правильного синтаксиса.
Выполняется следующий запрос:
SELECT t.Index1, t.Index2, COUNT( m.EventType )
FROM Table t
LEFT JOIN MEvents m ON
(m.Index1 = t.Index1 AND
m.Index2 = t.Index2 AND
(m.EventType = 'A' OR m.EventType = 'B')
)
WHERE (t.SpecialEventCount IS NULL)
GROUP BY t.Index1, t.Index2
Создает список триплетов Index1, Index2, EventCounts.
Это делается только для случая, когда t.SpecialEventCount имеет значение NULL. Запросы обновления, которые я пытаюсь записать, должны установить этот SpecialEventCount в этот счет, то есть COUNT (m.EventType) в запросе выше. Это число может быть 0 или любое положительное число (следовательно, левое соединение). Index1 и Index2 вместе являются уникальными в таблице t, и они используются для идентификации событий в MEvent.
Как мне изменить запрос выбора, чтобы он стал запросом обновления? То есть что-то вроде
UPDATE Table SET SpecialEventCount=COUNT(m.EventType).....
но я смущен тем, что положить туда и провалился с множеством различных догадок.
Ответы
Ответ 1
Я полагаю, что (Index1, Index2)
является уникальным ключом на Table
, иначе я бы ожидал, что ссылка на t.SpecialEventCount
приведет к ошибке.
Отредактированный запрос для использования подзапроса, поскольку он не работал с помощью GROUP BY
UPDATE
Table AS t
LEFT JOIN (
SELECT
Index1,
Index2,
COUNT(EventType) AS NumEvents
FROM
MEvents
WHERE
EventType = 'A' OR EventType = 'B'
GROUP BY
Index1,
Index2
) AS m ON
m.Index1 = t.Index1 AND
m.Index2 = t.Index2
SET
t.SpecialEventCount = m.NumEvents
WHERE
t.SpecialEventCount IS NULL
Ответ 2
Выполнение левого соединения с подзапросом приведет к созданию гигантского
временная таблица в памяти, которая не будет иметь индексов.
Для обновлений попробуйте избегать объединений и использовать коррелированные
вместо этого:
UPDATE
Table AS t
SET
t.SpecialEventCount = (
SELECT COUNT(m.EventType)
FROM MEvents m
WHERE m.EventType in ('A','B')
AND m.Index1 = t.Index1
AND m.Index2 = t.Index2
)
WHERE
t.SpecialEventCount IS NULL
Сделайте некоторые профилирования, но в некоторых случаях это может быть значительно быстрее.
Ответ 3
мой пример
update card_crowd as cardCrowd
LEFT JOIN
(
select cc.id , count(1) as num
from card_crowd cc LEFT JOIN
card_crowd_r ccr on cc.id = ccr.crowd_id
group by cc.id
) as tt
on cardCrowd.id = tt.id
set cardCrowd.join_num = tt.num;