一、ORDER BY返回的是游标而不是集合
SQL的理论其实是集合论,常见的类似求数据的交集、并集、差集都可以使用集合的思维来求解。
集合中的行之间没有预先定义的顺序,它只是成员的一种逻辑组合,成员之间的顺序无关紧要。
如下图,每一个括号里的内容就是一条记录,在没排序前,他们都是随机分布在集合中。
Student集合
但是对于带有排序作用的ORDER BY子句的查询,它返回的是一个对象,其中的行按特定的顺序组织在一起,我们把这种对象称为游标。
如下图,经过对Student表的ID进行ORDER BY排序后,Student表变成了有序对象,也就是我们上面说的游标。
Student对象
二、ORDER BY子句是唯一能重用列别名的一步
这里涉及SQL语句的语法顺序和执行顺序了,我们常见的SQL语法顺序如下:
SELECT DISTINCT < Top Num> <select list>
FROM [left_table]
<join_type> JOIN <right_table>
ON <join_condition>
WHERE <where_condition>
GROUP BY <group_by_list>
WITH <CUBE | RollUP>
HAVING <having_condition>
ORDER BY <order_by_list>
而数据库引擎在执行SQL语句并不是从SELECT开始执行,而是从FROM开始,具体执行顺序如下(关键字前面的数字代表SQL执行的顺序步骤):
( 8 ) SELECT ( 9 ) DISTINCT ( 11 )< Top Num> <select list>
( 1 ) FROM [left_table]
( 3 )<join_type> JOIN <right_table>
( 2 ) ON <join_condition>
( 4 ) WHERE <where_condition>
( 5 ) GROUP BY <group_by_list>
( 6 ) WITH <CUBE | RollUP>
( 7 ) HAVING <having_condition>
( 10 ) ORDER BY <order_by_list>
从上面可以看到SELECT在HAVING后才开始执行,这个时候SELECT后面列的别名只对后续的步骤生效,而对SELECT前面的步骤是无效的。所以如果你在WHERE,GROUP BY,或HAVING后面使用列的别名均会报错。
我们举例测试一下。
示例表Customers结构及数据如下:
1、WHERE后面不使用别名的情况
SELECT 姓名 AS Name, 地址 AS Address, 城市 AS City FROM Customers WHERE 城市='广州'
结果如下:
2、WHERE后面使用列别名的情况
SELECT 姓名 AS Name, 地址 AS Address, 城市 AS City FROM Customers WHERE City='广州'
执行结果如下:
从返回的消息中我们可以看到,重命名后的City并不能被WHERE识别,所以才会报“列名'City'无效”的提示。
其他关键字大家也可以使用上述方法进行测试,下面我们测试GROUP BY和HAVING后面使用列别名的情况。
3、测试GROUP BY后使用列别名
SELECT 城市 AS City FROM Customers GROUP BY City
结果如下:
4、测试HAVING后使用列别名
SELECT 城市 AS City FROM Customers GROUP BY 城市 HAVING COUNT(City)>1
结果如下: