Mysql中limit分页大偏移量的原因分析与优化

Mysql中limit分页大偏移量的原因分析与优化

文章内容截取掘金作者:Planeswalker23文章
原作者地址连接:https://juejin.cn/post/6862530465472741384

场景:
平时写的分页查询:(sex是索引列,非线上sql,只是为了分析)
select * from gen_studentinfo where sex = 1 limit s t a r t , {start}, start,{end}

  数据量少时,性能没有什么影响。随着数据量变大,start值的增加,执行时间会越来越长。

原因分析:

  1. 由于sex列是非聚簇索引,索引树中都存储的是主键id的值。查询所有的值,会发生一个回表的情况。在命中sex索引树中值为1的数据后,拿着它叶子节点上的值(主键id的值)去主键索引树上查询这一行其他列的值,最后返回到结果集中,这样第一行数据就查询成功。

  2. 最后这句sql要求limit
    start,end,也就是查询第start行到end行的数据。但是mysql会查询前start行,然后将前start行抛弃,最后结果集中就只剩下第start行到end行,执行结束。

    综上所述,造成limit大量便宜执行时间便就的原因有:
    1.查询多有列导致回表
    2.limit start,end会查询前start + end条数据,然后丢弃前start条数据。
    mysql花费大量时间在回表上,而其中a次回表的结果又不会出现在结果集中,这才导致查询时间越来越长。

优化方案:

  1. 覆盖索引
    理由:减少回表次数。
    我们受限得到start+1到start + end条数据的id,然后再进行回表获取其他列数据,那么就减少start次回表操作,速度会快上不少。
    覆盖索引:及时从非主聚簇索引中就能查到的想要数据,而不需要通过回表从主键索引中查询其他列,能够现主提升性能。
    聚簇索引:将数据存储与索引放到了一块,找到索引也就找到了数据。
    非聚簇索引:索引的存储和数据的存储是分离的,也就是说找到了索引但没找到数据,需要根据索引上的值(主键)再次回表查询,非聚簇索引也叫做辅助索引。
    select * from gen_studentinfo a join (select id from gen_studentinfo where sex = 1 limit start,end) b on a.id = b.id;
  2. 条件过滤
    select * from gen_studentinfo where sex = 1 and id > (select id from gen_studentinfo where sex = 1 limit start,1) limit 10;
    使用这种方式优化是有条件的:主键id必须是有序的,在有序的条件下,也可以使用比如创建时间等其他字段来代替主键id,但是前提是这个字段是建立了索引。
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
MySQL偏移过大指的是在使用LIMIT语句时,偏移的值过大,导致查询效率降低。具体优化方案如下: 1. 使用索引:首先要确认查询语句的字段是否创建了索引,索引可以提高查询效率。在需要使用偏移查询时,尽使用覆盖索引,即查询语句只包含索引列。 2. 分页优化:尽避免在大表上使用偏移查询,因为MySQL在执行偏移查询时需要先定位到偏移位置,这个过程是逐行扫描的,效率较低。可以考虑使用分页查询,每次查询包含固定数的数据,例如每页查询100条记录。 3. 缓存查询结果:如果查询结果不会频繁变动,可以将查询结果缓存起来,避免每次查询都执行相同的查询操作。 4. 避免跨页查询:当查询偏移较大时,如果只需要获取部分数据,可以通过调整查询条件来避免跨页查询。例如,将偏移设置为某页第一条记录的ID,然后使用LIMIT语句查询该ID之后的固定数的数据。 5. 优化查询语句:检查并优化查询语句是否存在多余的排序、过滤条件等,可以通过调整查询字段和条件,减少查询的数据,提高查询效率。 6. 增加硬件资源:如果以上优化措施仍然不能满足需求,可以考虑增加服务器的硬件资源,例如增加内存、CPU等,提升MySQL的处理能力。 综上所述,优化MySQL偏移过大的方法包括使用索引、分页查询、缓存查询结果、避免跨页查询、优化查询语句以及增加硬件资源等,根据具体情况选择合适的方法进行优化

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值