转载链接:https://www.cnblogs.com/yeyuzhuanjia/p/16267558.html
一、编写查询 SQL 顺序
1.select
2.from
3.join on
4.where
5.group by
6.having
7.order by
8.limit
二、MySQL 执行顺序
– from—>on—>join—>where—>group by—>having+(聚合函数)—>select—>distinct—>UNION—>order by—>limit
– UNION 在 ORDER BY 之前
1.from
2.on
3.join
4.where
5.group by
6.having
7.select
8.distinct
9.order by
10.limit
三、MySQL 执行顺序理解
第一步:加载 from 子句的前两个表计算笛卡尔积,生成虚拟表 vt1;
第二步:筛选关联表符合 on 表达式的数据,保留主表,生成虚拟表 vt2;
第三步:如果使用的是外连接,执行 on 的时候,会将主表中不符合 on 条件的数据也加载进来,做为外部行
第四步:如果 from 子句中的表数量大于 2,则重复第一步到第三步,直至所有的表都加载完毕,更新 vt3;
第五步:执行 where 表达式,筛选掉不符合条件的数据生成 vt4;
第六步:执行 group by 子句。group by 子句执行过后,会对子句组合成唯一值并且对每个唯一值只包含一行,生成 vt5,。一旦执行 group by,后面的所有步骤只能得到 vt5 中的列 (group by 的子句包含的列) 和聚合函数。
第七步:执行聚合函数,生成 vt6;
第八步:执行 having 表达式,筛选 vt6 中的数据。having 是唯一一个在分组后的条件筛选,生成 vt7;
第九步:从 vt7 中筛选列,生成 vt8;
第十步:执行 distinct,对 vt8 去重,生成 vt9。其实执行过 group by 后就没必要再去执行 distinct,因为分组后,每组只会有一条数据,并且每条数据都不相同。
第十一步:对 vt9 进行排序,此处返回的不是一个虚拟表,而是一个游标,记录了数据的排序顺序,此处可以使用别名;
第十二步:执行 limit 语句,将结果返回给客户端