Как сделать этот запрос SQL быстрее? - PullRequest
0 голосов
/ 26 сентября 2011

У меня есть следующий запрос:

SELECT DISTINCT `movies_manager_movie`.`id`, 
                 `movies_manager_movie`.`title`, 
                 `movies_manager_movie`.`original_title`, 
                 `movies_manager_movie`.`synopsis`, 
                 `movies_manager_movie`.`keywords`, 
                 `movies_manager_movie`.`release_date`, 
                 `movies_manager_movie`.`rating`, 
                 `movies_manager_movie`.`poster_web_url`, 
                 `movies_manager_movie`.`has_poster`, 
                 `movies_manager_movie`.`number`, 
                 `movies_manager_movie`.`has_sources`, 
                 `movies_manager_movie`.`season_id`, 
                 `movies_manager_movie`.`created`, 
                 `movies_manager_movie`.`updated`, 
                 `movies_manager_moviecache`.`activity_name` 

FROM `movies_manager_movie` 

LEFT OUTER JOIN `movies_manager_moviecache` ON (`movies_manager_movie`.`id` = `movies_manager_moviecache`.`movie_id`) 

WHERE (`movies_manager_movie`.`has_sources` = 1  
       AND (`movies_manager_moviecache`.`team_member_id` IN (

          SELECT U0.`id` FROM `movies_manager_movieteammember` U0 
          INNER JOIN `movies_manager_movieteammemberactivity` U1 ON (U0.`id` = U1.`team_member_id`) 
          WHERE U1.`movie_id` = 3588 ) 
                AND `movies_manager_movie`.`number` IS NULL
       ) 

    AND NOT (`movies_manager_movie`.`id` = 3588 )) 
    ORDER BY `movies_manager_moviecache`.`activity_name` DESC LIMIT 3;

Этот запрос может занять до 3 секунд, и я очень удивлен, так как у меня есть индексы везде и не более 35 строк в каждой из моих таблиц MyIsam,используя последнюю версию MySQL.

Я кэшировал все, что мог, но я должен, по крайней мере, запускать его 20000 раз в день, что составляет примерно 16 часов ожидания загрузки.И я уверен, что ни один из моих пользователей (ни Google Bot) не оценит время ожидания 4 секунды для каждой загрузки страницы.

Что я могу сделать, чтобы сделать это быстрее?

Я думал оДублирование поля из фильма в фильм кэш, поскольку все назначение кэша фильмов уже состоит в денормализации в сложное объединение.

Я попытался встроить подзапрос в список идентификаторов, но он удивительным образом удвоил время запроса.

Таблицы:

+----------------+--------------+------+-----+---------+----------------+
| Field          | Type         | Null | Key | Default | Extra          |
+----------------+--------------+------+-----+---------+----------------+
| id             | int(11)      | NO   | PRI | NULL    | auto_increment | 
| title          | varchar(120) | NO   | UNI | NULL    |                | 
| original_title | varchar(120) | YES  |     | NULL    |                | 
| synopsis       | longtext     | YES  |     | NULL    |                | 
| keywords       | varchar(120) | YES  |     | NULL    |                | 
| release_date   | date         | YES  |     | NULL    |                | 
| rating         | int(11)      | NO   |     | NULL    |                | 
| poster_web_url | varchar(255) | YES  |     | NULL    |                | 
| has_poster     | tinyint(1)   | NO   |     | NULL    |                | 
| number         | int(11)      | YES  |     | NULL    |                | 
| season_id      | int(11)      | YES  | MUL | NULL    |                | 
| created        | datetime     | NO   |     | NULL    |                | 
| updated        | datetime     | NO   |     | NULL    |                | 
| has_sources    | tinyint(1)   | NO   |     | NULL    |                | 
+----------------+--------------+------+-----+---------+----------------+
+---------------------+--------------+------+-----+---------+----------------+
| Field               | Type         | Null | Key | Default | Extra          |
+---------------------+--------------+------+-----+---------+----------------+
| id                  | int(11)      | NO   | PRI | NULL    | auto_increment | 
| name                | varchar(120) | NO   | UNI | NULL    |                | 
| biography           | longtext     | YES  |     | NULL    |                | 
| birth_date          | date         | YES  |     | NULL    |                | 
| picture_web_url     | varchar(255) | YES  |     | NULL    |                | 
| allocine_link       | varchar(255) | YES  |     | NULL    |                | 
| created             | datetime     | NO   |     | NULL    |                | 
| updated             | datetime     | NO   |     | NULL    |                | 
| has_picture         | tinyint(1)   | NO   |     | NULL    |                | 
| biography_linkyfied | longtext     | YES  |     | NULL    |                | 
+---------------------+--------------+------+-----+---------+----------------+

+----------------+--------------+------+-----+---------+----------------+
| Field          | Type         | Null | Key | Default | Extra          |
+----------------+--------------+------+-----+---------+----------------+
| id             | int(11)      | NO   | PRI | NULL    | auto_increment | 
| movie_id       | int(11)      | NO   | MUL | NULL    |                | 
| tag_slug       | varchar(100) | YES  | MUL | NULL    |                | 
| team_member_id | int(11)      | YES  | MUL | NULL    |                | 
| cast_rank      | int(11)      | YES  |     | NULL    |                | 
| activity_name  | varchar(30)  | YES  | MUL | NULL    |                | 
+----------------+--------------+------+-----+---------+----------------+

Mysql говорит мне, что это медленный запрос:

# Query_time: 3  Lock_time: 0  Rows_sent: 9  Rows_examined: 454128

Ответы [ 2 ]

1 голос
/ 27 сентября 2011

Попробуйте это:

SELECT `movies_manager_movie`.`id`, 
    `movies_manager_movie`.`title`, 
    `movies_manager_movie`.`original_title`, 
    `movies_manager_movie`.`synopsis`, 
    `movies_manager_movie`.`keywords`, 
    `movies_manager_movie`.`release_date`, 
    `movies_manager_movie`.`rating`, 
    `movies_manager_movie`.`poster_web_url`, 
    `movies_manager_movie`.`has_poster`, 
    `movies_manager_movie`.`number`, 
    `movies_manager_movie`.`has_sources`, 
    `movies_manager_movie`.`season_id`, 
    `movies_manager_movie`.`created`, 
    `movies_manager_movie`.`updated`, 
    (
        SELECT `movies_manager_moviecache`.`activity_name` 
        FROM `movies_manager_moviecache` 
        WHERE (`movies_manager_movie`.`id` = `movies_manager_moviecache`.`movie_id` 
            AND (`movies_manager_moviecache`.`team_member_id` IN (
                SELECT U0.`id` FROM `movies_manager_movieteammember` U0 
                    INNER JOIN `movies_manager_movieteammemberactivity` U1 ON (U0.`id` = U1.`team_member_id`) 
                WHERE U1.`movie_id` = 3588 ) 
            AND `movies_manager_movie`.`number` IS NULL
        ) ) LIMIT 1) AS `activity_name` 
FROM `movies_manager_movie`     
WHERE (`movies_manager_movie`.`has_sources` = 1  
    AND NOT (`movies_manager_movie`.`id` = 3588 )) 
ORDER BY `activity_name`  DESC 
LIMIT 3;

Дайте мне знать, как это работает

1 голос
/ 26 сентября 2011

Переместите movies_manager_movieteammemberactivity и movies_manager_movieteammember к основному оператору соединения (так что вы делаете левую внешнюю связь между movies_manager_movie и продуктом внутреннего соединения трех других таблиц).Это должно значительно ускорить ваш запрос.

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