limit offset慢查询背后的原因与解法

问题

问题起源于一个涉及到数据遍历的脚本。

该脚本会对一个MySQL表中的数据进行有条件的全表遍历。SQL如下:

select * from table where update_time < CURDATE() order by update_time desc limit 100 offset 10000;

这样写看起来很正常,但实际在数据量大了之后,使用起来开始出现问题,越来越慢,慢到不可接受,甚至影响其他的读写操作。

分析

原因就是limit offset这个语句,并不如人们望文生义想的那样,直接定位到第10000位然后取后面的100条记录。

而是令人发指的先一直一条一条读取到10100条,然后再根据offset的设置,舍弃前10000条记录,返回后面的100条记录。

其实原因也好理解,MySQL的数据存储并不是一个数组,可以直接根据下标获取第X位。即使给你搜索的字段加了索引,也只是使用该字段的值去建立一个新的二叉树(索引二叉树),来方便你快速找到数据位置。

但是试想一下,当你要在二叉树中找到第n大的数时,你并不能像找一个具体的值一样利用二叉树的能力快速找到,因为你也不知道每个节点的左子树和右子树分别有多少记录。

因此只能借用索引二叉树是个B+树这一特点,去利用叶子节点上的链表,去遍历你要数的所有节点。

这还不止。

MySQL不仅仅会让你遍历一遍索引值,我们知道MySQL默认的InnoDB引擎分为主键索引二叉树和辅助索引二叉树,你使用其他自己定义的索引时,只是得到主键,真正取数据还得根据索引得到的主键,去主键索引二叉树获取到具体的数据。

那此时,实际上你不仅在无效遍历前10000个索引节点,MySQL还会让你去根据遍历到的这10000个无效索引节点去真正地查10000次数据,这就是10000次无效的数据查询。

为什么MySQL一定要让你去查这些无效数据呢?因为MySQL的实现分为引擎层和数据层,limit offset只能作用于引擎层返回的结果集,因此对引擎层来说,他也不知道前10000个是会扔掉的数据,只能先一股脑地往上传。

更进一步的,为什么MySQL不把limit offset直接传给引擎层呢?是因为查询语句实际是由一个个算子组合起来的,比如有选择算子(where条件)、连接算子(join)、投影算则(select的字段)、数据源等,不同的算子有计算顺序,导致底层的算子是不知道上层计算条件的。

总得来说,这种实现就导致,数据量越大,offset得越多,速度就会越慢,对MySQL的压力就会越大。

解法

知道了问题根源之后,就可以对应地找解法。

解法1

比如我这里是要遍历数据,既然用offset遍历有性能问题,那就直接用主键id的范围条件来缩小范围。

select * from table where id > 10000 limit 100;

根据上面的分析我们可以指导,这样做,一方面直接省去了一次查询索引二叉树后再查主键二叉树的过程,而是直接就查主键二叉树并获取其节点上的数据。

另一方面,用大于的条件,从而利用好二叉树的特性,快速查找到数据的起始节点,然后获取其后的100条记录数据即可。

理解清楚,这和offset找第100001条节点的实现机制有本质区别。

不过如果此时使用explain对SQL性能进行检查,会发现rows的数量等于id > 10000后剩余的总记录数,而不是我们limit的100,比如总共如果有15000条记录,那此时的rows会是5000。

那这是否说明sql需要遍历id > 10000的所有记录呢?

不是的。explain得出的rows只是一个估算值。

实际上根据《MySQL EXPLAIN limits and errors》 一文所说,explain时,是不会考虑“LIMIT”的。

LIMIT is not taken into account while estimating number of rows

因此explain出的rows是id > 10000后剩余的总记录数,是符合预期的,而实际执行时,只会遍历到limit的数量就会结束了。

这种做法在20W的数据量级下,经过测试查询性能可以提升43倍。

解法2

上面的做法基本只适用于遍历的简单场景,从而可以直接使用主键去查询。

但大部分场景下,业务的查询都是附带条件的,也就是说必须要用到辅助的索引二叉树。

前面说了,如果用非主键的索引去遍历,会导致两次对二叉树的查询操作:先查索引二叉树找到节点的主键,再查主键索引二叉树取具体数据。

此时如果想实现一种条件下的翻页效果,直观可能会这样写SQL:

select * from table where update_time < CURDATE() limit 100 offset 10000;

此时MySQL经历的就是先根据条件找到10100条符合条件的记录(经过两个二叉树的查询),然后再抛弃前10000条。

那这里可以利用子查询不会真正获取数据的特性,进行优化:

select * from table where id in (select id from table where update_time < CURDATE()) limit 100 offset 10000;

注意这里子查询是根据辅助索引去查的,而主查询只根据了主键去查。

在子查询中并不会真正去访问主键索引二叉树获取数据,所以免去了10000次无效查询。

在子查询获取到id后,再用IN查询去在主键索引二叉树上遍历数据。

这种做法虽然也要查询10000条无用的数据,但由于是直接使用主键索引,所以比直接查询limit offset的做法会快两倍左右。

解法3

用IN操作,对于量大的情况始终不太优雅,因此还可以考虑用JOIN替代IN,自己JOIN自己:

select * from table as t1 inner join (select id from table where update_time < CURDATE() order by update_time desc limit 100 offset 10000) as t2 using (id);

这种做法经过测试会比最原始的SQL快10倍。

这里还需要注意的是,MySQL的JOIN有一个优化点,即用小表做驱动表去驱动大表。

比如对于 t1 left join t2 的情况,就建议把记录数较小的表放在前面,前面的表示驱动表,会扫描t1所有记录然后再去t2查询。

如果t1有M条记录,t2 N条,使用t2的索引的情况下,时间复杂度是M * logN左右,因此M的影响,也即t1的记录数对时间影响更大。

不过这里由于使用的是INNER JOIN,MySQL对INNER JOIN会自动使用小表,因此问题不大,实测下来耗时也相差无几。

更多解法

其实可以选择的解法还有很多,比如从业务层面限制要访问的数据,比如分表,比如其他奇诡的索引用法。

此外,这里介绍的解法,也更多地针对MySQL默认使用的InnoDB引擎去做优化,在不同的数据库存储引擎下,可能会有其他更合适的解法。


关注我的公众号【月亮与二进制】,鹅厂程序员的敲码间隙,也能读书观影练剑写字,分享给你我的世界

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容