MySQL объединяет непоследовательную производительность в зависимости от того, ГДЕ - PullRequest
1 голос
/ 02 ноября 2011

У меня есть запрос, выполнение которого занимает несколько секунд для определенных условий WHERE.Насколько мне известно, все проиндексировано правильно, но производительность все еще остается низкой.Все таблицы InnoDB, blog_users и blog_users_profiles содержат около 5 миллионов строк, а forum_contacts содержит около 10 тыс. Строк.

Запрос

SELECT blog.*
FROM blog_users AS blog
INNER JOIN blog_user_profiles AS profile ON blog.user_id = profile.user_id
INNER JOIN forum_contacts AS contact ON profile.forum_id = contact.user_id
WHERE blog.comments > :comment_cutoff
AND blog.last_active > :time_cutoff
AND contact.type = :contact_type
AND contact.area = :location
LIMIT 0, 100

Объяснить вывод

+----+-------------+---------+--------+------------------------------+-----------+---------+------------------------------+---------+-------------+
| id | select_type | table   | type   | possible_keys                | key       | key_len | ref                          | rows    | Extra       |
+----+-------------+---------+--------+------------------------------+-----------+---------+------------------------------+---------+-------------+
|  1 | SIMPLE      | blog    | range  | PRIMARY,last_active,comments | comments  | 3       | NULL                         | 1313813 | Using where |
|  1 | SIMPLE      | profile | eq_ref | PRIMARY,forum_id             | PRIMARY   | 3       | xc_db.blog.user_id           |       1 |             |
|  1 | SIMPLE      | contact | eq_ref | PRIMARY,type,area,user_id    | PRIMARY   | 80      | xc_db.profile.forum_id,const |       1 | Using where |
+----+-------------+---------+--------+------------------------------+-----------+---------+------------------------------+---------+-------------+

Структура таблицы(нерелевантные строки пропущены)

blog_users

+-------------+-----------------------+------+-----+---------+----------------+
| Field       | Type                  | Null | Key | Default | Extra          |
+-------------+-----------------------+------+-----+---------+----------------+
| user_id     | mediumint(8) unsigned | NO   | PRI | NULL    | auto_increment |
| username    | varchar(16)           | NO   | UNI | NULL    |                |
| comments    | mediumint(7) unsigned | NO   | MUL | NULL    |                |
| last_active | int(10) unsigned      | NO   | MUL | NULL    |                |
+-------------+-----------------------+------+-----+---------+----------------+

blog_user_profiles

+----------+-----------------------+------+-----+---------+-------+
| Field    | Type                  | Null | Key | Default | Extra |
+----------+-----------------------+------+-----+---------+-------+
| user_id  | mediumint(8) unsigned | NO   | PRI | NULL    |       |
| forum_id | mediumint(8) unsigned | NO   | MUL | 0       |       |
+----------+-----------------------+------+-----+---------+-------+

forum_contacts

+---------+-----------------------+------+-----+---------+-------+
| Field   | Type                  | Null | Key | Default | Extra |
+---------+-----------------------+------+-----+---------+-------+
| user_id | mediumint(8) unsigned | NO   | PRI | NULL    |       |
| type    | varchar(25)           | NO   | PRI | NULL    |       |
| area    | varchar(255)          | NO   | MUL | NULL    |       |
+---------+-----------------------+------+-----+---------+-------+

Странно то, что время выполнения кажетсябыть обратно связанным с общим количеством возвращаемых строк.Например, запрос типа contact.area = 'United States', который возвращает несколько тысяч строк, выполняется менее чем за 0,01 секунды.Однако запросы с меньшим количеством строк результатов, таких как contact.area = 'Egypt', который возвращает 20 строк, завершаются за 5 секунд.Нужно ли переписать мой запрос или есть проблема с индексами?

...