1

次のクエリは、CakePHP の i18n テーブルを使用しており、実行に約 1000 ミリ秒かかります。私の i18n テーブルは大きく、約 600,000 行あります。Cake スキーマに従って、このテーブルにデフォルトのインデックスがあります。

このようなクエリを高速化するにはどうすればよいですか? 可能な場合はキャッシュしていますが、キャッシュをクリアする必要がある場合、これらのクエリが実行され、パフォーマンスが著しく低下します。

SELECT `Category`.`id`, `Category`.`slug`, `Category`.`name`, `Category`.`description`, `Category`.`head_title`, `Category`.`display_title`, `Category`.`category_count`, `Category`.`product_count`, `Category`.`parent_id`, `Category`.`lft`, `Category`.`rght`, `Category`.`sort_order`, `Category`.`created`, `Category`.`modified`, `Category`.`image_processed`, ((rght - lft) = 1) AS `Category__is_child`, (`I18n__name`.`content`) AS `Category__i18n_name`, (`I18n__description`.`content`) AS `Category__i18n_description`, (`I18n__slug`.`content`) AS `Category__i18n_slug`, (`I18n__head_title`.`content`) AS `Category__i18n_head_title`, (`I18n__meta_description`.`content`) AS `Category__i18n_meta_description`, (`I18n__heading_title`.`content`) AS `Category__i18n_heading_title` FROM `fracmerl_dev`.`categories` AS `Category`
    INNER JOIN `fracmerl_dev`.`i18n` AS `I18n__name` 
        ON (`Category`.`id` = `I18n__name`.`foreign_key`
        AND `I18n__name`.`model` = 'Category' 
        AND `I18n__name`.`field` = 'name'
        AND `I18n__name`.`locale` = 'eng')
    INNER JOIN `fracmerl_dev`.`i18n` AS `I18n__description`
        ON (`Category`.`id` = `I18n__description`.`foreign_key`
        AND `I18n__description`.`model` = 'Category'
        AND `I18n__description`.`field` = 'description' 
        AND `I18n__description`.`locale` = 'eng')
    INNER JOIN `fracmerl_dev`.`i18n` AS `I18n__slug`
        ON (`Category`.`id` = `I18n__slug`.`foreign_key`
        AND `I18n__slug`.`model` = 'Category'
        AND `I18n__slug`.`field` = 'slug'
        AND `I18n__slug`.`locale` = 'eng')
    INNER JOIN `fracmerl_dev`.`i18n` AS `I18n__head_title`
        ON (`Category`.`id` = `I18n__head_title`.`foreign_key`
        AND `I18n__head_title`.`model` = 'Category'
        AND `I18n__head_title`.`field` = 'head_title'
        AND `I18n__head_title`.`locale` = 'eng')
    INNER JOIN `fracmerl_dev`.`i18n` AS `I18n__meta_description`
        ON (`Category`.`id` = `I18n__meta_description`.`foreign_key`
        AND `I18n__meta_description`.`model` = 'Category'
        AND `I18n__meta_description`.`field` = 'meta_description'
        AND `I18n__meta_description`.`locale` = 'eng')
    INNER JOIN `fracmerl_dev`.`i18n` AS `I18n__heading_title`
        ON (`Category`.`id` = `I18n__heading_title`.`foreign_key`
        AND `I18n__heading_title`.`model` = 'Category'
        AND `I18n__heading_title`.`field` = 'heading_title'
        AND `I18n__heading_title`.`locale` = 'eng')
WHERE `I18n__slug`.`content` = 'category-slug'
ORDER BY `Category`.`sort_order` ASC, `Category`.`name` ASC
LIMIT 1

編集 - インデックス、結果SHOW INDEX FROM i18n

i18n    0   PRIMARY 1   id  A   598455  NULL    NULL        BTREE
i18n    1   locale  1   locale  A   5   NULL    NULL        BTREE
i18n    1   model   1   model   A   3   NULL    NULL        BTREE
i18n    1   row_id  1   foreign_key A   18701   NULL  NULL  BTREE
i18n    1   field   1   field   A   9   NULL    NULL        BTREE
4

2 に答える 2

1

i18n テーブルが大きい場合は、分割して複数の変換テーブルを使用してみてください。

于 2013-07-18T07:16:03.903 に答える
1

どの結合テーブルWHEREから条件を移行してみてください。ON通常、違いはありませんが、クエリ オプティマイザーが正しく機能しないか効果がない場合は、以前の条件で行の量が減るため、パフォーマンスが向上します。

いくつかのリンク:

言うのを忘れました、もちろん、私はあなたができることを意味します

`I18n__slug`.`content` = 'category-slug' 

ON ステートメントに移動します。

何かのようなもの:

INNER JOIN `fracmerl_dev`.`i18n` AS `I18n__slug`
    ON (
    `I18n__slug`.`content` = 'category-slug'
    AND `Category`.`id` = `I18n__slug`.`foreign_key`
    AND `I18n__slugs`.`model` = 'Category'
    AND `I18n__slug`.`field` = 'slug'
    AND `I18n__slug`.`locale` = 'eng')
于 2013-07-18T07:56:32.497 に答える