Mysql
 sql >> Base de Dados >  >> RDS >> Mysql

Consulta lenta com cláusula HAVING - posso acelerá-la?


Uma solução rápida seria filtrar na subconsulta:
SELECT count(d.id) AS dcount, s.id, s.name
FROM sites s
LEFT JOIN deals d ON (s.id = d.site_id AND d.is_active = 1)
WHERE (s.is_active = 1)
AND s.id IN(
    SELECT g.site_id
    FROM gstats g
    WHERE g.start_date > '2015-04-30' AND g.site_id = s.id
    GROUP BY g.site_id
    HAVING SUM(g.results) > 100
)
GROUP BY s.id
ORDER BY dcount ASC

Caso contrário, você faz essa consulta de agrupamento para todos os candidatos possíveis. Podemos tornar isso mais elegante com EXISTS :
SELECT count(d.id) AS dcount, s.id, s.name
FROM sites s
LEFT JOIN deals d ON (s.id = d.site_id AND d.is_active = 1)
WHERE (s.is_active = 1)
AND EXISTS (
    SELECT 1
    FROM gstats g
    WHERE g.site_id = s.id AND g.start_date > '2015-04-30'
    HAVING SUM(g.results) > 100
)
GROUP BY s.id
ORDER BY dcount ASC

Mas ainda não terminamos, agora vamos usar o EXISTS para todos elemento. Isso é estranho, pois a consulta depende apenas de s.id , então depende apenas do grupo , não as linhas individuais. Portanto, um potencial speedup, mas isso depende do tamanho das tabelas, etc. é mover a condição para um HAVING declaração:
SELECT count(d.id) AS dcount, s.id, s.name
FROM sites s
LEFT JOIN deals d ON (s.id = d.site_id AND d.is_active = 1)
WHERE (s.is_active = 1)
GROUP BY s.id
ORDER BY dcount ASC
HAVING EXISTS (
    SELECT 1
    FROM gstats g
    WHERE g.site_id = s.id AND g.start_date > '2015-04-30'
    HAVING SUM(g.results) > 100
)