Подтвердить что ты не робот

Запрос обновления 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).....

но я смущен тем, что положить туда и провалился с множеством различных догадок.

4b9b3361

Ответ 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;