任何人都可以在这里帮助我。如何提高查询性能

时间:2012-12-27 05:05:52

标签: mysql query-performance

我的查询运行时间超过30分钟。即使它包含索引也是一个简单的查询。我们无法找到为什么它占用了太多的执行时间并影响了我们的整个数据库性能。   昨天它跑了:122.6分钟 任何人都可以在这里帮助我。如何提高查询性能

This is my query:
SELECT tab1.customer_id,tab1.row_mod,tab1.row_create,tab1.event_id,tab1.event_type,
tab1.new_value,tab1.old_value FROM tab1 force index (tab1_n2)where customer_id >= 1 and customer_id
< 5000000  and  (tab1.row_mod >= '2012-10-01') or  (tab1.row_create >=  '2012-10-01'  and tab1.row_create <  '2012-10-13');

Explain plan

+----+-------------+------------------+------+---------------------+------+---------+------+----------+-------------+
| id | select_type | table            | type | possible_keys       | key  | key_len | ref  | rows     | Extra       |
+----+-------------+------------------+------+---------------------+------+---------+------+----------+-------------+
|  1 | SIMPLE      | tab1 | ALL  | tab1_n2 | NULL | NULL    | NULL | 18490530 | Using where |
+----+-------------+------------------+------+---------------------+------+---------+------+----------+-------------+
1 row in set (0.00 sec)

Table structure:

mysql> show create table tab1\G
*************************** 1. row ***************************
       Table: tab1
Create Table: CREATE TABLE `tab1` (
  `customer_id` int(11) NOT NULL,
  `row_mod` datetime DEFAULT NULL,
  `row_create` datetime DEFAULT NULL,
  `event_id` int(11) DEFAULT NULL,
  `event_type` varchar(45) DEFAULT NULL,
  `new_value` varchar(255) DEFAULT NULL,
  `old_value` varchar(255) DEFAULT NULL,
  KEY `customer_id1` (`customer_id`),
  KEY `new_value_n1` (`new_value`),
  KEY `tab1_n1` (`row_create`),
  KEY `tab1_n2` (`row_mod`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8
1 row in set (0.00 sec)

请帮助我如何调整它。即使它有索引

2 个答案:

答案 0 :(得分:0)

可能是因为您使用的索引没有意义。

row_mod条件只是OR条件的一个分支,因此索引在这里没有多大帮助。如果强制每次查找索引而不删除任何行,则可能比全表扫描慢很多。好的经验法则是索引应该消除超过90%的行。

尝试不使用“强制索引”部分。

答案 1 :(得分:0)

尝试使用两种条件的UNION。这样每个条件都可以使用索引。

ALTER TABLE tab1 ADD INDEX idx_row_mod_customer_id (row_mod, customer_id);

ALTER TABLE tab1 ADD INDEX idx_row_create (row_create);

SELECT tab1.customer_id, tab1.row_mod, tab1.row_create, tab1.event_id, tab1.event_type,
tab1.new_value, tab1.old_value
FROM tab1
WHERE customer_id >= 1 and customer_id
< 5000000 AND tab1.row_mod >= '2012-10-01'

UNION

SELECT tab1.customer_id, tab1.row_mod, tab1.row_create, tab1.event_id, tab1.event_type,
tab1.new_value, tab1.old_value
FROM tab1
WHERE tab1.row_create >= '2012-10-01' AND tab1.row_create <  '2012-10-13';

为了进一步优化,您可以将所有选定列添加到两个索引,从而使MySQL不必将行加载到内存中。这将大大增加索引的大小,从而大大增加它们的内存需求。