Оптимизация Mysql для запроса select с предложением IN() внутри предложения where (поясните вывод)
У меня есть этот запрос:-
SELECT SUM(DISTINCT( ttagrels.id_tag IN ( 1816, 2642, 1906, 1398,
2436, 2940, 1973, 2791, 1389 ) )) AS
key_1_total_matches,
IF(( od.id_od > 0 ), COUNT(DISTINCT( od.id_od )), 0) AS
tutor_popularity,
td.*,
u.*
FROM tutor_details AS td
JOIN users AS u
ON u.id_user = td.id_user
JOIN all_tag_relations AS ttagrels
ON td.id_tutor = ttagrels.id_tutor
LEFT JOIN learning_packs AS lp
ON ttagrels.id_lp = lp.id_lp
LEFT JOIN learning_packs_categories AS lpc
ON lpc.id_lp_cat = lp.id_lp_cat
LEFT JOIN learning_packs_categories AS lpcp
ON lpcp.id_lp_cat = lpc.id_parent
LEFT JOIN learning_pack_content AS lpct
ON ( lp.id_lp = lpct.id_lp )
LEFT JOIN webclasses AS wc
ON ttagrels.id_wc = wc.id_wc
LEFT JOIN learning_packs_categories AS wcc
ON wcc.id_lp_cat = wc.id_wp_cat
LEFT JOIN learning_packs_categories AS wccp
ON wccp.id_lp_cat = wcc.id_parent
LEFT JOIN order_details AS od
ON td.id_tutor = od.id_author
LEFT JOIN orders AS o
ON od.id_order = o.id_order
WHERE ( u.country = 'IE'
OR u.country IN ( 'INT' ) )
AND u.status = 1
AND CASE
WHEN ( lp.id_lp > 0 ) THEN lp.id_status = 1
AND lp.published = 1
AND lpcp.status = 1
AND ( lpcp.country_code = 'IE'
OR lpcp.country_code IN ( 'INT' )
)
ELSE 1
END
AND CASE
WHEN ( wc.id_wc > 0 ) THEN wc.wc_api_status = 1
AND wc.id_status = 1
AND wc.wc_type = 0
AND
wc.class_date > '2010-06-16 11:44:40'
AND wccp.status = 1
AND ( wccp.country_code = 'IE'
OR wccp.country_code IN ( 'INT' )
)
ELSE 1
END
AND CASE
WHEN ( od.id_od > 0 ) THEN od.id_author = td.id_tutor
AND o.order_status = 'paid'
AND CASE
WHEN ( od.id_wc > 0 ) THEN od.can_attend_class = 1
ELSE 1
END
ELSE 1
END
AND ( ttagrels.id_tag IN ( 1816, 2642, 1906, 1398,
2436, 2940, 1973, 2791, 1389 ) )
GROUP BY td.id_tutor
HAVING key_1_total_matches = 1
ORDER BY tutor_popularity DESC,
u.surname ASC,
u.name ASC
LIMIT 0, 20
Числа внутри IN() на самом деле являются идентификаторами другой таблицы, называемой тегами, которая соответствует ключевым словам поиска, введенным пользователями. В этом примере пользователь искал "класс".
Смотрите вывод объяснения этого запроса здесь:- http://www.test.examvillage.com/Screenshot.png
Время, затраченное на этот запрос, составляет 0,0536 сек.
Но если количество значений в ttagrels.id_tag в () увеличивается (когда пользователь вводит больше ключевых слов для поиска), время выполнения возрастает примерно до 1-5 секунд и более. Например, если пользователь выполнил поиск "класс, доступный для преподавателей, и студенты 3 раза в день
время выполнения 4.2226 сек. Вывод запроса объяснения для этого запроса содержит 2513 строк.
Всего в таблице All_Tag_Relations 6,152 записей. Возможна ли дальнейшая оптимизация?
1 ответ
Я не знаю твоих данных, поэтому не могу дать однозначного ответа.
Я заметил в вашем запросе ключевое слово Distinct дважды SUM(DISTINCT( и COUNT(DISTINCT(
Это означает, что ваш оператор select возвращает несколько строк с одинаковым id_tag или id_od. Это означает, что ваш промежуточный результат перед группировкой содержит много строк. Вы должны сузить предложения where, чтобы ограничить количество строк.
Вставьте count(*) в ваш запрос и посмотрите количество строк. Если он больше ожидаемого, это объясняет вашу проблему.