[英]Lots of temporary tables in MySQL
我正在运行的网站下面有两个非常重复的查询。 在MySQL中,innoDB进程可以看到它们花费大量时间,并且每当我看到那里的进程时,它们就会继续创建临时表,并且要花很长时间执行并占用大量内存和CPU。
他们真的很糟糕,我以某种方式设法优化了这些。但是我无法做到这一点。
$getmoddetails = "SELECT a.id, a.name, a.defvar, a.description, a.icon, a.thumb, a.average_rating, a.total_rating, c.rating, a.group_access, d.long_name, a.editor_id, e.users_count
FROM dir_cat_item AS b
INNER JOIN dir_item AS a ON a.id = b.item_id
AND a.status = 'O'
LEFT JOIN dir_item_notation_user_map AS c ON a.id = c.item_id
AND c.user_id =%u
LEFT JOIN users AS d ON d.id = a.editor_id
LEFT JOIN (SELECT item_id, COUNT(*) AS users_count
FROM module
GROUP BY item_id) AS e ON e.item_id = b.item_id
WHERE a.id=%u";
$getnbModules_by_col = "SELECT
posx,COUNT(posx) as nb
FROM module WHERE
user_id = %u
AND profile_id = %u
GROUP BY posx
ORDER BY posx ASC";
模块上的表索引
- item_id - user_id - profile_id - uniq
对于用户表
- id - username
任何建议请...
更新:-
CREATE TABLE IF NOT EXISTS `module` (
`item_id` mediumint(8) unsigned NOT NULL DEFAULT '0',
`user_id` int(10) unsigned NOT NULL DEFAULT '0',
`profile_id` int(3) unsigned NOT NULL DEFAULT '0',
`posx` tinyint(3) unsigned NOT NULL DEFAULT '0',
`posy` tinyint(3) unsigned NOT NULL DEFAULT '0',
`posj` tinyint(3) unsigned NOT NULL DEFAULT '0',
`x` smallint(5) unsigned NOT NULL DEFAULT '0',
`y` smallint(5) unsigned NOT NULL DEFAULT '0',
`typ` char(1) CHARACTER SET utf8 NOT NULL DEFAULT 'D',
`variables` text COLLATE utf8_unicode_ci,
`uniq` smallint(5) unsigned NOT NULL DEFAULT '1',
`blocked` tinyint(1) unsigned NOT NULL DEFAULT '0',
`minimized` tinyint(1) unsigned NOT NULL DEFAULT '0',
`old_id` tinyint(3) unsigned DEFAULT NULL,
`feed_id` mediumint(8) unsigned NOT NULL DEFAULT '0',
`shared` varchar(33) COLLATE utf8_unicode_ci DEFAULT NULL,
`currentview` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL,
KEY `item_id` (`item_id`,`user_id`,`profile_id`,`uniq`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
item_id 18 A No
user_id 393467 A No
profile_id 393467 A No
uniq 393467 A No
先感谢您
为了使第二个查询获得最佳性能,您需要在module
表上有适当的索引,例如
... ON module (user_id, profile_id, posx)
对于第一个查询,您可能会受益于module
表上的其他索引:
... ON module (item_id)
但是,如果没有表定义,列的数据类型和基数,就不可能提出明确的建议。
在您的第一个查询中,建议您在内联视图(派生表)中添加别名为e
的谓词。 我认为MySQL不会将谓词从外部查询推入内联视图。
( SELECT item_id
, COUNT(*) AS users_count
FROM module
WHERE item_id = %u
GROUP BY item_id
) AS e
您将需要在WHERE子句中提供与在外部查询的WHERE子句中提供的值相同的值。 从我在那儿看的书...
e.item_id = b.item_id = a.id = %u
通过在该内联视图中添加该WHERE子句,应该可以减少从模块表中检索到的行数,并且该派生表将只有一行。 前导item_id
索引将是覆盖索引。 EXPLAIN PLAN应该显示正在Using index
而不要显示正在Using filesort
。
如果您的第一个查询要回退相对较少的行,则可以考虑在模块表上使用相关子查询,而不是在第一个查询中连接到派生表(别名为e
)(以避免实现大派生)表)。 (但是,通常,关联的子查询可能是真正的性能杀手。但是在某些情况下,外部查询会拉回少量的行,子查询的重复执行实际上比生成大型派生表要好。您只需要从中几行即可。)
SELECT a.id
, a.name
, a.defvar
, a.description
, a.icon
, a.thumb
, a.average_rating
, a.total_rating
, c.rating
, a.group_access
, d.long_name
, a.editor_id
, ( SELECT SUM(1)
FROM module e
WHERE e.item_id = b.item_id
AND e.item_id = %u
) AS users_count
FROM dir_cat_item b
JOIN dir_item a
ON a.id = b.item_id
AND b.item_id = %u
AND a.status = 'O'
LEFT
JOIN dir_item_notation_user_map c
ON c.item_id = a.id
AND c.item_id = %u
AND c.user_id = %u
LEFT
JOIN users d
ON d.id = a.editor_id
WHERE a.id = %u
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.