mysql left join_MySQL中 Order By 和 Limit 的排序问题

在写一个分页查询记录的sql时,要根据添加的时间逆序分页输出,之前的写法是酱紫:

select    record.a,    y.cfrom    (        select            a,b        from            x        order by timestamp desc        limit 0,10    ) recordleft join yon record.b = y.d;

因为一些新的需求,要在后面加一些where条件,limit操作不能在嵌套查询里面加了,于是乎把limit 0,10提出来放到最外面,结果order by还留在里面。

当时想嵌套查询出来的record表已经按timestamp字段逆序排列了,再left另一张表,最终再limit出来的结果应该也是逆序的,但结果却很打脸,是正序的。

首先控制变量,代码回滚到之前,把后来加的各种逻辑都去掉,还原到上述sql,只把limit 0,10移到最后,发现timestamp是正序的,那么问题应该就出在这里了,与后来加的其他逻辑没有关系。

那么再试一下删掉limit操作,结果timestamp是无序的!

这不可能啊,于是认真看了下数据,发现一些规律,可能是按y表的自增id或created_at时间字段排序的(因为这两个字段是索引字段),那么到这里,至少可以得到一个简单的结论,就是联表查询结果,不是按照嵌套查询中的order by排序的,现在正向一看,确实不可能按这个排序,因为括号里面的逻辑对括号外是不可见的。

还有个问题,上述去掉limit后,最终不是按left join主表的顺序输出,按照常理想象,mysql是循环主表的记录去关联另一张表,那么输出的顺序应该还是主表的顺序啊,但结果却是按另一张表的字段排序的,这又是为什么呢?

去官方手册中找找线索,发现order by模块中有这么一句话。

08c52c6ab523ef23a11dcbf217d51311.png

再去limit模块中看一下

98f22579ebe351e45ef8fba133e1163c.png

从以上两个截图中,我们可以发现一些端倪,limit操作会对查询有一些优化,查询到指定条数的数据,就可以提前结束了,比如我们本文中的left操作,拿到10条结果就结束查询线程,返回客户端。

猜测,如果没有limit操作,反正全部都要join,可能mysql会对循环逻辑做一些优化,不一定要按主表来循环,思想类似于java编译中的重排序,也对应了上面截图中的那句话。

采用最简单、最粗暴的方式,直接把order by 和 limit操作放到最外面就ok啦,其实效率上并没有什么降低,只要索引建的合理即可。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
根据引用的解释,方案二的子查询在MySQL会被自动优化成直接的左连接,因此在执行explain命令时并没有出现差异。然而,在子查询加入了having子句后,MySQL无法自动优化成直接的左连接,因此按照SQL语句的调用顺序执行,导致explain结果出现了一个额外的子查询行。这样实现了原本的目的,即从10,000 x 10,000 x 10,000的复杂度降低到10,000 x 10,000的复杂度,从而提升了速度。 根据引用的方案二的描述,通过将两次左连接合并为一次来减少笛卡尔积的操作,可以显著提升速度。具体的SQL结构如下: ``` select p.id,p.name,pss.sort from table1 p left join ( select name,sort from table2 ps left join table2 ps2 on ps.name = ps2.name and ... where ... ) pss on p.name = pss.name where ... order by pss.sort asc,p.sale desc,p.time desc limit 0,10 ``` 而引用提到的第三种解决方案是将SQL语句拆分开来,因为是左连接,关联关系可以分开执行,然后逐个执行,观察执行情况。 因此,根据以上三个引用的内容,可以得出解决MySQL左连接特别慢的三种方法,即通过优化子查询、合并左连接操作和拆分SQL语句来提升速度。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [mysql 优化慢复杂sql (多个left join 数量过大 order by 巨慢)](https://blog.csdn.net/qq_40835969/article/details/128239398)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *3* [mysql left join 查询慢的问题排查](https://blog.csdn.net/tianjiliuhen/article/details/127446340)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值