在Mysql中我们常常用order by来进行排序,使用limit来进行分页,当需要先排序后分页时我们往往使用类似的写法select * from 表名 order by 排序字段 limt M,N。但是这种写法却隐藏着较深的使用陷阱。在排序字段有数据重复的情况下,会很容易出现排序结果与预期不一致的问题。

比如现在有一张user表,表结构及数据如下:

表结构

表数据

现在想根据创建时间升序查询user表,并且分页查询,每页2条,那很容易写出sql为:select * from user order by create_time limit pageNo,2;

在执行查询过程中会发现:
1、查询第一页数据时:

第一页查询结果

2、查询第四页数据时:

第四页查询结果

user表共有8条数据,有4页数据,但是实际查询过程中第一页与第四页竟然出现了相同的数据。

这是什么情况?难道上面的分页SQL不是先将两个表关联查询出来,然后再排好序,再取对应分页的数据吗???

上面的实际执行结果已经证明现实与想像往往是有差距的,实际SQL执行时并不是按照上述方式执行的。这里其实是Mysql会对Limit做优化,具体优化方式见官方文档:https://dev.mysql.com/doc/refman/5.7/en/limit-optimization.html
这个是5.7版本的说明,提取几个问题直接相关的点做下说明。

Paste_Image.png

上面官方文档里面有提到如果你将Limit row_count与order by混用,mysql会找到排序的row_count行后立马返回,而不是排序整个查询结果再返回。如果是通过索引排序,会非常快;如果是文件排序,所有匹配查询的行(不带Limit的)都会被选中,被选中的大多数或者全部会被排序,直到limit要求的row_count被找到了。如果limit要求的row_count行一旦被找到,Mysql就不会排序结果集中剩余的行了。

这里我们查看下对应SQL的执行计划:

Paste_Image.png

可以确认是用的文件排序,表确实也没有加额外的索引。所以我们可以确定这个SQL执行时是会找到limit要求的行后立马返回查询结果的。

不过就算它立马返回,为什么分页会不准呢?

官方文档里面做了如下说明:

Paste_Image.png

如果order by的字段有多个行都有相同的值,mysql是会随机的顺序返回查询结果的,具体依赖对应的执行计划。也就是说如果排序的列是无序的,那么排序的结果行的顺序也是不确定的。

基于这个我们就基本知道为什么分页会不准了,因为我们排序的字段是create_time,正好又有几个相同的值的行,在实际执行时返回结果对应的行的顺序是不确定的。对应上面的情况,第一页返回的name为8的数据行,可能正好排在前面,而第四页查询时name为8的数据行正好排在后面,所以第四页又出现了。

那这种情况应该怎么解决呢?

官方给出了解决方案:

Paste_Image.png

如果想在Limit存在或不存在的情况下,都保证排序结果相同,可以额外加一个排序条件。例如id字段是唯一的,可以考虑在排序字段中额外加个id排序去确保顺序稳定。

所以上面的情况下可以在SQL再添加个排序字段,比如fund_flow的id字段,这样分页的问题就解决了。修改后的SQL可以像下面这样:
SELECT * FROM user ORDER BY create_time,id LIMIT 6,2;

再次测试问题解决!!

Mysql order by与limit混用陷阱相关推荐

  1. java混合分页_坑,MySQL中 order by 与 limit 混用,分页会出现问题!

    在Mysql中我们常常用order by来进行排序,使用limit来进行分页,当需要先排序后分页时我们往往使用类似的写法select * from 表名 order by 排序字段 limt M,N. ...

  2. 坑,MySQL中 order by 与 limit 混用,分页会出现问题!

    作者:丘八老爷 blog.csdn.net/qiubabin/article/details/70135556 在Mysql中我们常常用order by来进行排序,使用limit来进行分页,当需要先排 ...

  3. 如何解决MySQL order by limit语句的分页数据重复问题?

    文章来源:https://www.jianshu.com/p/544c319fd838 0 问题描述 在MySQL中我们通常会采用limit来进行翻页查询,比如limit(0,10)表示列出第一页的1 ...

  4. layui分页limit不显示_小心避坑:MySQL分页时使用 limit+order by 会出现数据重复问题...

    20大进阶架构专题每日送达 来源:www.jianshu.com/p/544c319fd838 进入主题前先插一下,当当优惠码福利来一波!当当全场自营图书5折,用优惠码:J2JYFK(长按复制),满2 ...

  5. mysql scrapy 重复数据_小心避坑:MySQL分页时使用 limit+order by 会出现数据重复问题...

    作者:猿码道http://www.jianshu.com/p/544c319fd838 0 问题描述 在MySQL中我们通常会采用limit来进行翻页查询,比如limit(0,10)表示列出第一页的1 ...

  6. 小心避坑:MySQL分页时使用 limit+order by 会出现数据重复问题

    点击上方"Java知音",选择"置顶公众号" 技术文章第一时间送达! 作者:猿码道 www.jianshu.com/p/544c319fd838 0 问题描述 ...

  7. MySQL中 Order By 和 Limit 的排序问题

    点击上方蓝色"方志朋",选择"设为星标" 回复"666"获取独家整理的学习资料! 作者 | 春卷要炸着吃 来源 | www.cnblogs. ...

  8. MySQL中Order By与Limit不要一起用

    切记!MySQL 中 ORDER BY 与 LIMIT 不要一起用,有大坑...... 现象与问题 ORDER BY 排序后,用 LIMIT 取前几条,发现返回的结果集的顺序与预期的不一样. 下面是我 ...

  9. mysql div 没有小数,mysql order by limit 的一个坑

    mysql order by limit 的一个坑 分页查询的时候遇到的坑: 发现的问题: 对单个无索引的字段进行排序后limit .发现当被排序字段有相同值时并且在limit范围内,取的值并不是正常 ...

最新文章

  1. 华为云大数据存储的冗余方式是三副本_大数据入门:HDFS数据副本存放策略
  2. mysql5.7 only_full_group_by_MySQL5.7默认打开ONLY_FULL_GROUP_BY模式问题与解决方案
  3. Java基础学习总结(16)——Java制作证书的工具keytool用法总结
  4. linux搜索过滤器,Unix和Linux管道(pipe)和过滤器(filter)使用介绍
  5. 【52】写了placement new也要写placement delete
  6. service mesh 数据平面nginmesh
  7. 中点和中值滤波的区别_桥式整流二极管及滤波电容如何选择
  8. Ubuntu 14.04 文件服务器--samba的安装和配置
  9. 划重点:微信小程序1月9日正式上线
  10. Android 系统(261)---SIM卡不识或者掉卡简单分析
  11. MkDocs 文档编辑器
  12. ubuntu16 安装opencv
  13. 使用Visio2003更新SQL2005数据库
  14. 计算机自己创作游戏,说说自己做的数据库:我不爱玩电脑游戏,自己做数据库自己欣赏...
  15. matlab脉冲压缩,雷达线性调频脉冲压缩的原理及其matlab仿真
  16. 深度卷机网络(Deep CNNs)的GFLOPS与参数量计算
  17. 小程序实现书籍翻页效果
  18. 如何给华硕笔记本在光驱位加装另一块linux系统固态硬盘?
  19. 代码圈复杂度治理小结
  20. 热感觉、热舒适、热满意度、热需求与热偏好

热门文章

  1. 10种常用降维算法源代码(python)
  2. 网易干货 | 浅析视频Jitter Buffer
  3. Think as developer, 从深入理解业务实现框架开始
  4. ICML 2018大奖出炉:伯克利、MIT获最佳论文,复旦大学榜上有名
  5. 区分各浏览器的CSS hack(包括360、搜狗、opera)
  6. 两大上市公司联手,打造智慧城市安防领域生态闭环
  7. webbench 代码阅读
  8. 人人都是 DBA(X)资源信息收集脚本汇编
  9. $#,$@,$0,$1,$2,$*,$$,$?,${}
  10. javascript 编程思想