当前位置: 代码迷 >> Sql Server >> 关于SQL的执行顺序有关问题,求指点
  详细解决方案

关于SQL的执行顺序有关问题,求指点

热度:44   发布时间:2016-04-24 18:33:25.0
关于SQL的执行顺序问题,求指点
我的数据库中有一个表 两个字段
CREATE TABLE [dbo].[Test](
[ID] [int] NOT NULL,
[Time] [varchar](50) NOT NULL

测试数据:
INSERT [dbo].[Test] ([ID], [Time]) VALUES (1, N'标题11')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (2, N'标题22')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (3, N'标题33')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (4, N'2014-02-25 15:40:32')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (5, N'2014-02-25 15:40:49')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (6, N'2014-02-25 15:40:56')

我的查询语句:
SELECT ID,TIMES 
FROM (SELECT ID,TIME FROM Test 
WHERE ISDATE(Time)=1) AS T(ID,TIMES)
WHERE DATEDIFF(YYYY, convert(date,TIMES), GETDATE()) = 0 

我将FROM后的子查询放在临时表中,select临时表where datediff(..)没有问题
但整个语句执行的时候就会报错:

消息 241,级别 16,状态 1,第 1 行
从字符串转换日期和/或时间时,转换失败。


我查了一下SQL的执行顺序,先执行的FROM,然后执行where,既然我的FROM已经将非日期的数据过滤掉了,在WHERE的时候判断为什么还会报转换失败的错误,求大侠指点迷津..
------解决方案--------------------
修改成这样试试:
--drop table test

CREATE TABLE [dbo].[Test](
[ID] [int] NOT NULL,
[Time] [varchar](50) NOT NULL


INSERT [dbo].[Test] ([ID], [Time]) VALUES (1, N'标题11')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (2, N'标题22')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (3, N'标题33')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (4, N'2014-02-25 15:40:32')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (5, N'2014-02-25 15:40:49')
INSERT [dbo].[Test] ([ID], [Time]) VALUES (6, N'2014-02-25 15:40:56')
go


SELECT ID,TIMES 
FROM (SELECT ID,TIME FROM Test 
WHERE ISDATE(Time)=1) AS T(ID,TIMES)
WHERE left(TIMES,4)= DATENAME(YYYY,GETDATE()) 
/*
ID TIMES
4 2014-02-25 15:40:32
5 2014-02-25 15:40:49
6 2014-02-25 15:40:56
*/

------解决方案--------------------
SQL 的语法并不按照语法顺序执行

SQL 语句有一个让大部分人都感到困惑的特性,就是:SQL 语句的执行顺序跟其语句的语法顺序并不一致。SQL 语句的语法顺序是:

SELECT[DISTINCT]
FROM
WHERE
GROUP BY
HAVING
UNION
ORDER BY
为了方便理解,上面并没有把所有的 SQL 语法结构都列出来,但是已经足以说明 SQL 语句的语法顺序和其执行顺序完全不一样,就以上述语句为例,其执行顺序为:

FROM
WHERE
GROUP BY
HAVING
SELECT
DISTINCT
UNION
ORDER BY
关于 SQL 语句的执行顺序,有三个值得我们注意的地方:

1、 FROM 才是 SQL 语句执行的第一步,并非 SELECT 。数据库在执行 SQL 语句的第一步是将数据从硬盘加载到数据缓冲区中,以便对这些数据进行操作。(译者注:原文为“The first thing that happens is loading data from the disk into memory, in order to operate on such data.”,但是并非如此,以 Oracle 等常用数据库为例,数据是从硬盘中抽取到数据缓冲区中进行操作。)

2、 SELECT 是在大部分语句执行了之后才执行的,严格的说是在 FROM 和 GROUP BY 之后执行的。理解这一点是非常重要的,这就是你不能在 WHERE 中使用在 SELECT 中设定别名的字段作为判断条件的原因。

1
2
3
SELECT A.x + A.y AS z
FROM A
WHERE z = 10 -- z 在此处不可用,因为SELECT是最后执行的语句!
如果你想重用别名z,你有两个选择。要么就重新写一遍 z 所代表的表达式:

1
2
3
SELECT A.x + A.y AS z
FROM A
WHERE (A.x + A.y) = 10
…或者求助于衍生表、通用数据表达式或者视图,以避免别名重用。请看下文中的例子。

3、 无论在语法上还是在执行顺序上, UNION 总是排在在 ORDER BY 之前。很多人认为每个 UNION 段都能使用 ORDER BY 排序,但是根据 SQL 语言标准和各个数据库 SQL 的执行差异来看,这并不是真的。尽管某些数据库允许 SQL 语句对子查询(subqueries)或者派生表(derived tables)进行排序,但是这并不说明这个排序在 UNION 操作过后仍保持排序后的顺序。

注意:并非所有的数据库对 SQL 语句使用相同的解析方式。如 MySQL、PostgreSQL和 SQLite 中就不会按照上面第二点中所说的方式执行。

我们学到了什么?

既然并不是所有的数据库都按照上述方式执行 SQL 预计,那我们的收获是什么?我们的收获是永远要记得: SQL 语句的语法顺序和其执行顺序并不一致,这样我们就能避免一般性的错误。如果你能记住 SQL 语句语法顺序和执行顺序的差异,你就能很容易的理解一些很常见的 SQL 问题。

当然,如果一种语言被设计成语法顺序直接反应其语句的执行顺序,那么这种语言对程序员是十分友好的,这种编程语言层面的设计理念已经被微软应用到了 LINQ 语言中。
  相关解决方案