[Z]T-SQL查询进阶--深入理解子查询
时间:2022-03-13 22:56
SQL有着非常强大且灵活的查询方式,而多表连接操作往往也可以用子查询进行替代,本篇文章将会讲述子查询的方方面面。
简介
子查询本质上是嵌套进其他SELECT,UPDATE,INSERT,DELETE语句的一个被限制的SELECT语句,在子查询中,只有下面几个子句可以使用
- SELECT子句(必须)
- FROM子句(必选)
- WHERE子句(可选)
- GROUP BY(可选)
- HAVING(可选)
- ORDER BY(只有在TOP关键字被使用时才可用)
子查询也可以嵌套在其他子查询中,这个嵌套最多可达32层。子查询也叫内部查询(Inner query)或者内部选择(Inner Select),而包含子查询的查询语句也叫做外部查询(Outter)或者外部选择(Outer Select),子查询的概念可以简单用下图阐述:
上图是作为数据源使用的一个子查询.
通常来讲,子查询按照子查询所返回数据的类型,可以分为三种,分别为:
- 返回一张数据表(Table)
- 返回一列值(Column)
- 返回单个值(Scalar)
下面,我们按照这三种方式来阐述子查询
子查询作为数据源使用
当子查询在外部查询的FROM子句之后使用时,子查询被当作一个数据源使用,即使这时子查询只返回一个单一值(Scalar)或是一列值(Column),在这里依然可以看作一个特殊的数据源,即一个二维数据表(Table).作为数据源使用的子查询很像一个View(视图),只是这个子查询只是临时存在,并不包含在数据库中。
比如这个语句:
P.ProductID, P.Name, P.ProductNumber, M.Name ProductModelName Production.Product P ( Name, ProductModelID Production.ProductModel) M P.ProductModelID = M.ProductModelID
上述子查询语句将ProductModel表中的子集M,作为数据源(表)和Product表进行内连接。结果如下:
作为数据源使用也是子查询最简单的应用。当然,当子查询作为数据源使用时,也分为相关子查询和无关子查询,这会在文章后面介绍到.
子查询作为选择条件使用
作为选择条件的子查询也是子查询相对最复杂的应用.
作为选择条件的子查询是那些只返回一列(Column)的子查询,如果作为选择条件使用,即使只返回单个值,也可以看作是只有一行的一列.比如:
在AdventureWorks中:
我想取得总共请病假天数大于68小时的员工:
[FirstName] ,[MiddleName] ,[LastName] [AdventureWorks].[Person].[Contact] ContactID ( EmployeeID [AdventureWorks].[HumanResources].[Employee] SickLeaveHours>68)
结果如下:
如何区别相关子查询和无关子查询呢?最简单的办法就是直接看子查询本身能否执行,比如上面的例子中的子查询:
( * [AdventureWorks].[HumanResources].[Employee] e c.ContactID=e.ContactID e.SickLeaveHours>68)
这一句本身执行本身会报错.因为这句引用到了外部查询的表
对于无关子查询来说,整个查询的过程为子查询只执行一次,然后交给外部查询,比如:
* AdventureWorks.HumanResources.Employee SickLeaveHours> SQLRESULT
上面的无关子查询,整个查询过程可以看作是子查询首先返回SQLResult(SQL结果集),然后交给外部查询使用,整个过程子查询只执行一次
而相反,作为相关子查询,子查询的执行的次数依赖于外部查询,外部查询每执行一行,子查询执行一次,比如:
还是上面的例子:我想取得总共请病假天数大于68天的员工
[FirstName] ,[MiddleName] ,[LastName] [AdventureWorks].[Person].[Contact] c ( * [AdventureWorks].[HumanResources].[Employee] e c.ContactID=e.ContactID e.SickLeaveHours>68) ---- step 1: [FirstName] ,[MiddleName] ,[LastName] [AdventureWorks].[Person].[Contact] c ( * [AdventureWorks].[HumanResources].[Employee] e 1=e.ContactID e.SickLeaveHours>68) ---- step 2: [FirstName] ,[MiddleName] ,[LastName] [AdventureWorks].[Person].[Contact] c ( * [AdventureWorks].[HumanResources].[Employee] e 2=e.ContactID e.SickLeaveHours>68) ---- step n: [FirstName] ,[MiddleName] ,[LastName] [AdventureWorks].[Person].[Contact] c ( * [AdventureWorks].[HumanResources].[Employee] e n=e.ContactID e.SickLeaveHours>68)
如上面代码所示。上面的相关子查询实际上会执行N次(N取决与外部查询的行数),外部查询每执行一行,都会将对应行所用的参数传到子查询中,如果子查询有对应值,则返回TRUE(既当前行被选中并在结果中显示),如果没有,则返回FALSE。然后重复执行下一行。
子查询作为计算列使用
当子查询作为计算列使用时,只返回单个值(Scalar) 。用在SELECT语句之后,作为计算列使用。同样分为相关子查询和无关子查询
相关子查询的例子比如:我想取得每件产品的名称和总共的销量
[Name], ( (*) AdventureWorks.Sales.SalesOrderDetail S S.ProductID=P.ProductID) SalesAmount [AdventureWorks].[Production].[Product] P
部分结果如下:
当子查询作为计算列使用时,会针对外部查询的每一行,返回唯一的值。
同样的,SQL子查询都可以使用其他语句达到同样的效果,上面的语句和如下语句达到同样的效果:
P.Name,(S.ProductID) [AdventureWorks].[Production].[Product] P AdventureWorks.Sales.SalesOrderDetail S S.ProductID=P.ProductID P.Name
子查询作为计算列且作为无关子查询时使用,只会一次性返回但一值,这里就不再阐述了。
小结
本篇文章通过子查询的三种不同用途来阐述子查询。同时,所有的子查询还可以分为相关子查询和无关子查询,而子查询所实现的功能都可以使用连接或者其他方式实现。但一个好的作家应该是掌握丰富的词汇,而不是仅仅能表达出自己的意思。学会多种SQL查询方式是学习SQL查询必经之路。
[Z]T-SQL查询进阶--深入理解子查询,布布扣,bubuko.com