У меня такой запрос:
SELECT
*,
(
SELECT COUNT(DISTINCT client_id)
FROM 1097_course_students_tbl
WHERE course_cycl_id=id
AND stts_id <> 8
AND client_id IN(SELECT id FROM 1097_clients_tbl WHERE is_removed=0)
) AS cnt
FROM 1097_course_cycle_tbl
WHERE (course_id IN (SELECT id FROM 1097_courses_tbl WHERE is_removed=2))
ORDER BY start_date DESC
Мне нужно сделать его более эффективным, потому что он занимает слишком много времени
какие-либо предложения?
Спасибо
Попробуйте следующее
SELECT cc.*,IFNULL(q.cnt,0) cnt
FROM 1097_course_cycle_tbl cс
JOIN 1097_courses_tbl с ON c.id=cc.course_id AND c.is_removed=2
LEFT JOIN
(
SELECT cs.course_cycl_id,COUNT(DISTINCT cs.client_id) cnt
FROM 1097_course_students_tbl cs
JOIN 1097_clients_tbl c ON cs.client_id=c.id AND c.is_removed=0
WHERE cs.stts_id<>8
GROUP BY cs.course_cycl_id
) q
ON q.course_cycl_id=cс.id
ORDER BY cc.start_date DESC
Я думаю, что id
в 1097_courses_tbl
и 1097_clients_tbl
является первичным ключом. Поэтому я заменил IN
в JOIN
.
И я преобразовал подзапрос из блока SELECT
который был выполнен для каждой строки в подзапрос с GROUP BY
который используется в LEFT JOIN
. Здесь он будет выполняться только один раз и вернуть всю необходимую информацию.
IN
вJOIN
а окончательный запрос будет таким же. Я предпочитаю использоватьJOIN
если это возможно. Но коррелированный подзапрос в блокеSELECT
очень часто не оптимален. Я думаю, что в вашем запросе это замедляет запрос.