Select字句在逻辑上是SQL语句最后进行处理的最后一步,所以,以下查询会发生错误:

SELECT   YEAR(OrderDate) AS OrderYear,   COUNT(DISTINCT CustomerID) 
AS NumCusts  FROM dbo.Orders  GROUP BY OrderYear;

因为group by是在Select之前进行的,那个时候orderYear这个列并没有形成。

如果要查询成功,可以像下面进行修改:

SELECT OrderYear, COUNT(DISTINCT CustomerID) AS NumCusts  
FROM (SELECT YEAR(OrderDate) AS OrderYear, CustomerID     FROM dbo.Orders) AS D 
 GROUP BY 
OrderYear;

还有一种很特殊的写法:

SELECT OrderYear, COUNT(DISTINCT CustomerID) AS NumCusts  FROM 
(SELECT YEAR(OrderDate), CustomerID     FROM dbo.Orders) AS D(OrderYear, CustomerID) 
 GROUP BY OrderYear;
 

在作者眼里,他是非常喜欢这种写法的,因为更清晰,更明确,更便于维护。

在查询中使用参数定向产生一批结果,这个技巧没有什么好说的。

嵌套查询,在处理逻辑上是从里向外进行执行的。

多重引用,有可能你的SQL语句包含了多次从一个表进行查询后进行连接组合。比如你要比较每年的顾客数同先前年的顾客数的变化,所以你的查询就必须JOIN了2个相同的表的实例,这也是不可避免的。

Common Table Expressions (CTE)

CTE是在SQL2005新加入的一种表的表示类型。

它的定义如下:

WITH cte_name

AS

(

cte_query

)

outer_query_refferring to_cte_name;

注意:因为在标准的T-SQL语言中已经包含了WITH AS,所以为了区分,CTE在语句的结尾加上了“;”作为停止符。

CTE实例一(结果集别名)

WITH C AS  (   
        SELECT YEAR(OrderDate) AS OrderYear, CustomerID   FROM dbo.Orders  
)  
SELECT OrderYear, COUNT(DISTINCT CustomerID) AS NumCusts  
FROM C  GROUP BY OrderYear;
 

当然,作者本人有更推荐的写法:

WITH C(OrderYear, CustomerID) AS  (   SELECT YEAR(OrderDate), CustomerID   FROM dbo.Orders  )  
SELECT OrderYear, COUNT(DISTINCT CustomerID) AS NumCusts  FROM C  GROUP BY OrderYear;

CTE实例二(多重CTEs)

WITH C1 AS  (   SELECT YEAR(OrderDate) AS OrderYear, CustomerID   FROM dbo.Orders  ),
  C2 AS  (   
SELECT OrderYear, COUNT(DISTINCT CustomerID) AS NumCusts   FROM C1  
 GROUP BY OrderYear  )  
SELECT OrderYear, NumCusts  FROM C2  WHERE NumCusts > 70;

CTE实例三(多重引用)

WITH YearlyCount AS  
(   SELECT YEAR(OrderDate) AS OrderYear,    COUNT(DISTINCT CustomerID) AS NumCusts   FROM dbo.Orders   
GROUP BY YEAR(OrderDate)  
)  
SELECT Cur.OrderYear,   Cur.NumCusts AS CurNumCusts, Prv.NumCusts AS PrvNumCusts,  
 Cur.NumCusts - Prv.NumCusts AS Growth  
FROM YearlyCount AS Cur   LEFT OUTER JOIN YearlyCount AS Prv    ON Cur.OrderYear = Prv.OrderYear + 1;
 

CTE实例四(修改数据)

1.把从customer表查询出来的结果,动态的组装进新表CustomersDups里:

IF OBJECT_ID('dbo.CustomersDups') IS NOT NULL   DROP TABLE dbo.CustomersDups;  
GO    
WITH CrossCustomers AS  
(   SELECT 1 AS c, C1.*   FROM dbo.Customers AS C1, dbo.Customers AS C2  ) 
 SELECT ROW_NUMBER() OVER(ORDER BY c) AS KeyCol,   CustomerID, CompanyName,
 ContactName, ContactTitle, Address,   City, Region, PostalCode, Country, Phone, Fax  
INTO dbo.CustomersDups  FROM CrossCustomers;  

2.使用CTE移除数据,只保留CustomerDups表里同一CustomerID里KeyCol为最大的记录。

WITH JustDups AS  (   SELECT * FROM dbo.CustomersDups AS C1   
WHERE KeyCol <    (SELECT MAX(KeyCol) FROM dbo.CustomersDups AS C2     
WHERE C2.CustomerID = C1.CustomerID)  )  DELETE FROM JustDups;
 

CTE实例五(对象容器)

即提供了封装的能力,有利于组件化的编程。作者额外的提醒,CTE无法直接内嵌,但是可以通过把CTE封装进一个对象容器里并从一个外部的CTE里对这容器的数据进行查询而实现内嵌。

作者也说明了,使用CTEs在VIEW和UDFs里是没有什么价值的。

有个例子,如下:

CREATE VIEW dbo.VYearCnt  AS  
WITH YearCnt AS  (   SELECT YEAR(OrderDate) AS OrderYear,   
 COUNT(DISTINCT CustomerID) AS NumCusts   FROM dbo.Orders   
GROUP BY YEAR(OrderDate)  ) 
 SELECT * FROM YearCnt;

CTE实例六(CTEs的递归)

作者给了一个例子,来讲述这个在SQL2005的新内容,CTEs的递归。

根据employeeId,返回此员工的信息,并包含所有下级员工的信息。(等级关系基于empolyeeId和reportsTo的属性)所返回的结果包含下列字段,employeeId,reportsTo,FirstName,LastName。

作者在这里,给予了一个最佳的索引方式:

CREATE UNIQUE INDEX idx_mgr_emp_ifname_ilname   ON dbo.Employees(ReportsTo, EmployeeID) 
  INCLUDE(FirstName, LastName);

作者的解释: 这个索引将通过一个单独的查询(局部扫描)来取得每个经理的直接下级。Include(FristName,LastName)加在这里,即是覆盖列。

小知识:什么Include索引?

Include索引是SQL2005的新功能。Include索引的列并不影响索引行的物理存储顺序,他们作为一个挂件‘挂在'索引行上。挂这些‘挂件'的目的在于,只需要扫描一把索引就获得了这些附加数据。

回到作者的例子上,下面是递归的代码:

WITH EmpsCTE AS  
(   SELECT EmployeeID, ReportsTo, FirstName, LastName   FROM dbo.Employees  
 WHERE EmployeeID = 5   
UNION ALL     
SELECT EMP.EmployeeID, EMP.ReportsTo, EMP.FirstName, EMP.LastName   
FROM EmpsCTE AS MGR    JOIN dbo.Employees AS EMP     ON EMP.ReportsTo = MGR.EmployeeID  )
  SELECT * FROM EmpsCTE;

理解:一个递归的CTE包含了至少2个查询,第一个查询在CTE的身体里类似于一格锚点。这个锚点仅仅返回一个有效的表,并作为递归的一个锚。从上的例子看出来,锚点仅仅返回了一个employeeID = 5 的一行。然后的第2个查询是作为递归成员。当查询到下属成员的结果为空时,此递归结束。

如果你担心递归会造成永久循环,你可以使用下面的表达:

WITH cte_name AS (cte_body) outer_query OPTION (MAXRECURSION n);

默认的n为100,当n=0时,无限制。

转载于:https://www.cnblogs.com/Qbit/archive/2012/12/11/T-SQL-with-as-%E5%85%B3%E9%94%AE%E5%AD%97.html

T-SQL with as 关键字相关推荐

  1. SQL LEFT JOIN 关键字

    SQL LEFT JOIN 关键字 LEFT JOIN 关键字会从左表 (table_name1) 那里返回所有的行,即使在右表 (table_name2) 中没有匹配的行. LEFT JOIN 关键 ...

  2. MySQL查询语种关键字_SQL——SQL语言全部关键字详解

    http://blog.csdn.net/quinnnorris/article/details/71056445 数据库中我们做常用的就是SQL基本查询语言,甚至有些人认为数据库就是SQL,SQL就 ...

  3. SQL JOIN\SQL INNER JOIN 关键字\SQL LEFT JOIN 关键字\SQL RIGHT JOIN 关键字\SQL FULL JOIN 关键字...

    SQL join 用于根据两个或多个表中的列之间的关系,从这些表中查询数据. Join 和 Key 有时为了得到完整的结果,我们需要从两个或更多的表中获取结果.我们就需要执行 join. 数据库中的表 ...

  4. SQL中各关键字以及DBeaver工具的使用

    SQL中各关键字以及DBeaver工具的使用 1. SQL执行顺序 2. 聚合函数 2.1 常用的聚合函数 2.2 group by ... having ... 2.3 Error 3. DBeav ...

  5. 【SQL】interval关键字用法

    [SQL]interval关键字使用-用法 取时间间隔,用于日期计算. 例如: 下面展示一些 应用代码. ##1.取最近29天的数据 select * from table where op_time ...

  6. 金仓数据库 KingbaseES PL/SQL 过程语言参考手册(19. D PL/SQL保留字和关键字)

    19. D PL/SQL保留字和关键字 保留字和关键字是在 PL/SQL 中具有特殊含义的标识符.它们不区分大小写.有关它们的更多信息,请参阅"保留字词和关键字". 表 19.1  ...

  7. 数据库关键字问题mysql 运行sql语句check关键字报错

    mysql 运行sql语句check关键字报错.出错:因为check为 mysql的保留关键字,改名即可. 遇到数据库语句问题可以参考下这个问题,之前的access数据库用的check作为表明,代码迁 ...

  8. oracle怎样删除关键字,SQL DROP CONSTRAINT 关键字

    SQL DROP CONSTRAINT 关键字 DROP CONSTRAINT DROP CONSTRAINT命令用于删除UNIQUE,PRIMARY KEY,FOREIGN KEY或CHECK约束. ...

  9. SQL系列ROLLUP关键字

    为什么要引入ROLLUP? 可方便的生成"合计"."小计"."总计" 等混合统计的报表 如下数据集A经过ROLLUP运算可生成数据集B 接下 ...

  10. 从线上慢sql看explain关键字

    大家好,我是烤鸭:     最近有点忙的头晕,又懒又累,正好线上遇到慢sql的问题,就说下 MySQL Explain 关键字的解析和使用示例. explain 关键字说明 使用explain关键字可 ...

最新文章

  1. 游戏开发--开源21---CroftSoft Code Library |Gibdo|JOGR...
  2. Android之网络丢包
  3. ABAP中的系统变量SY-INDEX与SY-TABIX
  4. NET问答: 重写了 Equals,还有必要重写 GetHashCode 吗?
  5. (数据库系统概论|王珊)第十章数据库恢复技术-第四、五、六、七节:数据库恢复技术和数据库镜像
  6. 【XLL 框架库函数】 Excel/Excel12f
  7. 如何在树莓派上安装Fedora 25
  8. Harbor 使用 Helm 一键安装
  9. HDFS体系结构详解
  10. 在线短视频缩略图剪切工具
  11. OMRON继电器基础讲解
  12. 【NOIP2012-Day2-T2-换教室】
  13. 大学计算机基础vfp程序设计课程试验报告簿,VF程序设计实验报告册(实践教程).doc...
  14. 2022蓝桥杯——砍竹子
  15. linux邮件客户端配置文件,在Deepin V20下配置Evolution邮件客户端,添加新邮箱全过程...
  16. 前后端分离项目全环境搭建(Ruoyi框架)
  17. js模拟表单html形式,JS模拟并美化的表单控件完整实例
  18. 习题 5.12 编写一程序,将两个字符串连接起来,结果取代第一个字符串。
  19. cl.wedid.us index.php,Increase in sudden death from coronary artery disease in young adults
  20. 下载的文件需要右键属性解除锁定的一次性解决方法

热门文章

  1. netbeans工具栏字体太小
  2. 决策树-熵计算-ID3算法(转)
  3. apt-get update时卡在 waiting for headers(转)
  4. matlab矩阵及其基本运算—特征值分解和svd奇异值分解
  5. 基于pip的安装lxml库报错解决方案
  6. python 决策树 math库 c45算法
  7. Python脚本监控线上AMQ Number of Pending Messages数量
  8. RouterOS SOCKS代理服务器(官方文档翻译)
  9. 【SICP练习】79 练习2.51
  10. 2.Strings and Console Output(字符串与输出)