数据库用的5.7.18,500多万条数据,uid有索引,出现个奇怪的问题:
如果ASC后 limit 4条没有问题
但是同样的语句如果 limit 1到3条,直接卡死
EXPLAIN这条语句又正常:
多次测试DESC没有此问题,只有使用了ASC且limit 1-3 范围内才会出现直接卡死情况,请问这种问题应该往哪个方向排查?
是否是mysql这个版本的bug这种可能性?
(后期多次测试发现是 limit=1 且和 ORDER BY组合的情况下才会出现随机选择索引,和DESC无关)
找到原因了,这就是mysql5.7的一个坑:
搭建了 5.7.24 和 5.7.18 多次还原测试,当 limit 等于1,uid 和 order by 的字段create_time 都存在索引,那么mysql就会随机性在 uid 和 create_time 选择一个索引查询,如果选择了 create_time 他就按照order by 排序全表扫,直到查到最近对应的uid一条数据,这样就变的很卡,不知道为什么第一次查他基本选择uid查,所以测试不会出现问题,但是后面多次查他又抛弃了uid使用了create_time扫。
测试:
CREATE TABLE `testsql` (
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT COMMENT 'ID',
`uid` int(10) unsigned NOT NULL COMMENT 'UID',
`pull_qr` varchar(50) NOT NULL COMMENT '任意数据',
`type` tinyint(1) DEFAULT NULL COMMENT '任意数据',
`check_status` tinyint(1) DEFAULT '0' COMMENT '状态:0未完成,1已完成',
`create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
PRIMARY KEY (`id`),
KEY `uid` (`uid`),
KEY `create_time` (`create_time`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='测试数据';
把这个表随机写入500万条数据,对应uid数据不会超过1万条,然后查询:
SELECT `id`,`pull_qr` FROM `testsql` WHERE `uid` = '1' AND `check_status` = 0 AND `type` = 1 ORDER BY `create_time` ASC LIMIT 1
第一次查基本会以UID为索引查,没问题,后面断开,再查,多次重复操作,他居然只走 create_time 索引扫了,总结就是当 possible_keys 为uid key 为 create_time 的时候,有一定概率会使用 create_time 排序后扫下来,直到扫到对应的一条uid才停止。
最终解决方案:force index(uid) 强制走指定索引