2010-10-06 82 views
1

我正在对具有唯一列的连接进行索引扫描;它声称正在检查大量的行,即使它只查找一行。在mysql查询计划中发生意外的索引扫描

这是查询:

select t.id, 
      t.twitter_id, 
      t.screen_name, 
      t.text  
     from tweets t 
inner join twitter_handle th on th.handle = t.screen_name 
    order by t.created_at desc 
    limit 1; 

添加/删除限制条款并不会改变查询计划。我希望它会扫描许多等于limit子句中的数字的tweets的created_at索引,然后针对twitter_handle执行eq_ref查找。

根据解释的查询计划,但是,是:

+----+-------------+-------+-------+---------------+-------------+---------+------+--------+----------------------------------------------+ 
| id | select_type | table | type | possible_keys | key   | key_len | ref | rows | Extra          | 
+----+-------------+-------+-------+---------------+-------------+---------+------+--------+----------------------------------------------+ 
| 1 | SIMPLE  | th | index | NULL   | handle  | 32  | NULL | 100126 | Using index; Using temporary; Using filesort | 
| 1 | SIMPLE  | t  | ref | screen_name | screen_name | 17  | func |  2 | Using where         | 
+----+-------------+-------+-------+---------------+-------------+---------+------+--------+----------------------------------------------+ 

注意号码在连接检查的索引扫描和ref = FUNC用于第二表的行的订购。

这个查询显示在我的缓慢的查询日志中,我很困惑为什么mysql选择这种方式执行查询。

的架构这两个表:

CREATE TABLE `twitter_handle` (
    `handle_id` int(11) NOT NULL AUTO_INCREMENT, 
    `handle` varchar(30) CHARACTER SET ascii NOT NULL, 
    `twitter_token_id` int(11) DEFAULT NULL, 
    `name` varchar(255) CHARACTER SET utf8 DEFAULT NULL, 
    `twitter_user_id` int(11) unsigned DEFAULT NULL, 
    `location` varchar(100) CHARACTER SET utf8 DEFAULT NULL, 
    `profile_image_url` varchar(255) CHARACTER SET utf8 DEFAULT NULL, 
    `followers_count` int(11) DEFAULT NULL, 
    `twitter_list_id` int(4) DEFAULT NULL, 
    `last_update` timestamp NULL DEFAULT CURRENT_TIMESTAMP, 
    `bio` varchar(160) CHARACTER SET utf8 DEFAULT NULL, 
    PRIMARY KEY (`handle_id`), 
    UNIQUE KEY `handle` (`handle`), 
    KEY `twitter_token_id` (`twitter_token_id`), 
    KEY `twitter_user_id` (`twitter_user_id`) 
) ENGINE=InnoDB; 

CREATE TABLE `tweets` (
    `id` int(11) NOT NULL AUTO_INCREMENT, 
    `twitter_id` char(15) DEFAULT NULL, 
    `screen_name` varchar(15) NOT NULL, 
    `logged_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, 
    `text` char(200) NOT NULL, 
    `created_at` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00', 
    `status` enum('pending','processed','ignored','pending_delete','deleted','pending_tweet','preview') NOT NULL DEFAULT 'pending', 
    `interaction_id` int(11) DEFAULT NULL, 
    PRIMARY KEY (`id`), 
    UNIQUE KEY `twitter_id_UNIQUE` (`twitter_id`), 
    UNIQUE KEY `interaction_id_idx` (`interaction_id`), 
    UNIQUE KEY `interaction_id` (`interaction_id`,`status`), 
    KEY `screen_name` (`screen_name`,`created_at`), 
    KEY `status_2` (`status`,`created_at`), 
    KEY `created_at_2` (`created_at`) 
) ENGINE=InnoDB; 
+0

用于表别名和丰富的信息 – 2010-10-06 21:40:17

回答

0

的原因是在twitter_handle手柄是ASCII字符集,但在SCREEN_NAME鸣叫是latin1的!