相关文章推荐
强健的企鹅  ·  Matlab App ...·  1 年前    · 
威武的灯泡  ·  c++ - UINT32_MAX 的 C ...·  1 年前    · 

适用于: SQL Server Azure SQL 数据库 Azure SQL 托管实例 Azure Synapse Analytics Analytics Platform System (PDW)

子查询是一个嵌套在 SELECT INSERT UPDATE DELETE 语句或其他子查询中的查询。

本文中的示例使用可在 AdventureWorks 示例数据库 中下载的 AdventureWorks2016 数据库。

任何允许使用表达式的地方都可以使用子查询。 在此示例中,子查询用作语句中名为 MaxUnitPrice 的 SELECT 列表达式。

USE AdventureWorks2016;
SELECT Ord.SalesOrderID, Ord.OrderDate,
    (SELECT MAX(OrdDet.UnitPrice)
     FROM Sales.SalesOrderDetail AS OrdDet
     WHERE Ord.SalesOrderID = OrdDet.SalesOrderID) AS MaxUnitPrice
FROM Sales.SalesOrderHeader AS Ord;

子查询基础知识

子查询也称为内部查询或内部选择,而包含子查询的语句也称为外部查询或外部选择。

许多包含子查询的 Transact-SQL 语句也可以表述为联接。 其他问题只能通过子查询提出。 在 Transact-SQL 中,包含子查询的语句和不包含子查询的语义等效版本之间通常没有性能差异。 有关SQL Server如何处理查询的体系结构信息,请参阅 SQL 语句处理。 但是,在一些必须检查存在性的情况中,使用联接会产生更好的性能。 否则,为确保消除重复值,必须为外部查询的每个结果都处理嵌套查询。 所以在这些情况下,联接方式会产生更好的效果。

以下示例演示返回相同结果集和执行计划的子查询 SELECT 和联接 SELECT

USE AdventureWorks2016;
/* SELECT statement built using a subquery. */
SELECT [Name]
FROM Production.Product
WHERE ListPrice =
    (SELECT ListPrice
     FROM Production.Product
     WHERE [Name] = 'Chainring Bolts' );
/* SELECT statement built using a join that returns
   the same result set. */
SELECT Prd1.[Name]
FROM Production.Product AS Prd1
     JOIN Production.Product AS Prd2
       ON (Prd1.ListPrice = Prd2.ListPrice)
WHERE Prd2.[Name] = 'Chainring Bolts';

嵌套在外部 SELECT 语句中的子查询包括以下组件:

  • 包含常规选择列表组件的常规 SELECT 查询。
  • 包含一个或多个表或视图名称的常规 FROM 子句。
  • 可选的 WHERE 子句。
  • 可选的 GROUP BY 子句。
  • 可选的 HAVING 子句。
  • 子查询的 SELECT 查询总是使用圆括号括起来。 它不能包含 COMPUTEFOR BROWSE 子句,并且只能在还指定 TOP 子句时包含 ORDER BY 子句。

    子查询可以嵌套在外部 SELECTINSERTUPDATEDELETE 语句的 WHEREHAVING 子句内,也可以嵌套在其他子查询内。 尽管根据可用内存和查询中其他表达式的复杂程度的不同,嵌套限制也有所不同,但嵌套到 32 层是可能的。 个别查询可能不支持 32 层嵌套。 任何可以使用表达式的地方都可以使用子查询,只要它返回的是单个值。

    如果表仅出现在子查询中,而未出现在外部查询中,则该表中的列不能包含在输出中, (外部查询) 的选择列表。

    包含子查询的语句通常采用以下格式中的一种:

  • WHERE expression [NOT] IN (subquery)
  • WHERE expression comparison_operator [ANY | ALL] (subquery)
  • WHERE [NOT] EXISTS (subquery)
  • 在某些 Transact-SQL 语句中,子查询的计算方法就像它是一个独立的查询一样。 从概念上讲,子查询结果被替换为外部查询 (尽管这不一定是SQL Server实际处理具有子查询) Transact-SQL 语句的方式。

    有三种基本的子查询。 它们是:

  • 在通过 IN 或由 ANYALL 修改的比较运算符引入的列表上操作。
  • 通过未修改的比较运算符引入且必须返回单个值。
  • 通过 EXISTS 引入的存在测试。
  • 子查询规则

    子查询受下列限制的制约:

  • 通过比较运算符引入的子查询选择列表只能包括一个表达式或列名称(对 EXISTS 执行的 IN 或对列表执行的 SELECT * 子查询除外)。
  • 如果外部查询的 WHERE 子句包括列名称,它必须与子查询选择列表中的列是联接兼容的。
  • ntexttextimage 数据类型不能在子查询的选择列表中使用。
  • 由于它们必须返回单个值,因此由未修改比较运算符引入的子查询 (不后跟关键字 (keyword) ANYALL) 不能包含 GROUP BYHAVING 子句。
  • 关键字 (keyword) DISTINCT 不能与包含 GROUP BY的子查询一起使用。
  • COMPUTE无法指定 和 INTO 子句。
  • 只有指定了 TOP 时才能指定 ORDER BY
  • 不能更新使用子查询创建的视图。
  • 按照惯例,由 EXISTS 引入的子查询的选择列表有一个星号 (*),而不是单个列名。 因为由 EXISTS 引入的子查询创建了存在测试并返回 TRUE 或 FALSE 而非数据,所以由 EXISTS 引入的子查询的规则与标准选择列表的规则相同。
  • 限定子查询中的列名称

    在下列示例中,外部查询的 WHERE 子句中的 BusinessEntityID 列是由外部查询的 FROM 子句中的表名 (Sales.Store) 隐性限定的。 对子查询的选择列表中 CustomerID 的引用则是由子查询的 FROM 子句(即通过 Sales.Customer 表)来限定的。

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Sales.Store
    WHERE BusinessEntityID NOT IN
        (SELECT CustomerID
         FROM Sales.Customer
         WHERE TerritoryID = 5);
    

    一般的规则是,语句中的列名通过同级 FROM 子句中引用的表来隐性限定。 如果子查询的 子句中引用 FROM 的表中不存在列,则外部查询的 子句中 FROM 引用的表将隐式限定该列。

    下面是指定了以下隐式假设的查询外观:

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Sales.Store
    WHERE Sales.Store.BusinessEntityID NOT IN
        (SELECT Sales.Customer.CustomerID
         FROM Sales.Customer
         WHERE TerritoryID = 5);
    

    显式声明表名称永远不会出错,并且始终可以重写有关具有显式限定的表名称的隐式假设。

    如果某个子查询中引用的列不存在于该子查询的 FROM 子句引用的表中,而存在于外部查询的 FROM 子句引用的表中,则该查询可以正确执行。 SQL Server使用外部查询中的表名隐式限定子查询中的列。

    子查询自身可以包括一个或多个子查询。 一个语句中可以嵌套任意数量的子查询。

    以下查询将查找作为销售人员的雇员的姓名。

    USE AdventureWorks2016;
    SELECT LastName, FirstName
    FROM Person.Person
    WHERE BusinessEntityID IN
        (SELECT BusinessEntityID
         FROM HumanResources.Employee
         WHERE BusinessEntityID IN
            (SELECT BusinessEntityID
             FROM Sales.SalesPerson)
    

    下面是结果集:

    LastName                                           FirstName
    -------------------------------------------------- -----------------------
    Jiang                                              Stephen
    Abbas                                              Syed
    Alberts                                            Amy
    Ansman-Wolfe                                       Pamela
    Campbell                                           David
    Carson                                             Jillian
    Ito                                                Shu
    Mitchell                                           Linda
    Reiter                                             Tsvi
    Saraiva                                            Jos
    Vargas                                             Garrett
    Varkey Chudukatil                                  Ranjit
    Valdez                                             Rachel
    Tsoflias                                           Lynn
    Pak                                                Jae
    Blythe                                             Michael
    Mensa-Annan                                        Tete
    (17 row(s) affected)
    

    最里层查询将返回销售人员的 ID。 再上一层查询将用这些销售人员 ID 进行取值,并返回雇员的联系 ID 号。 最后,外部查询将使用这些联系 ID 查找雇员的姓名。

    也可以将该查询表示为一个联接:

    USE AdventureWorks2016;
    SELECT LastName, FirstName
    FROM Person.Person c
    INNER JOIN HumanResources.Employee e
    ON c.BusinessEntityID = e.BusinessEntityID
    JOIN Sales.SalesPerson s
    ON e.BusinessEntityID = s.BusinessEntityID;
    

    相关子查询

    许多查询都可以通过执行一次子查询并将得到的值代入外部查询的 WHERE 子句中进行计算。 在包括相关子查询(也称为重复子查询)的查询中,子查询依靠外部查询获得值。 这意味着子查询是重复执行的,为外部查询可能选择的每一行均执行一次。 此查询在 SalesPerson 表中检索奖金为 5000 且雇员标识号与 Employee 和 SalesPerson 表中的标识号相匹配的雇员的名和姓的一个实例。

    USE AdventureWorks2016;
    SELECT DISTINCT c.LastName, c.FirstName, e.BusinessEntityID
    FROM Person.Person AS c JOIN HumanResources.Employee AS e
    ON e.BusinessEntityID = c.BusinessEntityID
    WHERE 5000.00 IN
        (SELECT Bonus
        FROM Sales.SalesPerson sp
        WHERE e.BusinessEntityID = sp.BusinessEntityID) ;
    

    下面是结果集:

    LastName FirstName BusinessEntityID
    -------------------------- ---------- ------------
    Ansman-Wolfe Pamela 280
    Saraiva José 282
    (2 row(s) affected)
    

    无法独立于外部查询来计算此语句中的上一个子查询。 它需要 Employee.BusinessEntityID 的值,但此值会随着SQL Server检查 Employee 中的不同行而更改。 这正是此查询的计算方式:SQL Server通过将每行中的值替换为内部查询来考虑 Employee 表的每一行以包含在结果中。 例如,如果 SQL Server 首先检查 的行,Syed Abbas则变量 Employee.BusinessEntityID 采用值 285,该值SQL Server替换到内部查询中。 这两个查询示例使用相关子查询表示上一个示例的分解。

    USE AdventureWorks2016;
    SELECT Bonus
    FROM Sales.SalesPerson
    WHERE BusinessEntityID = 285;
    

    结果是 0.00 (Syed Abbas 没有收到奖金,因为他们不是) 的销售人员,因此外部查询的计算结果为:

    USE AdventureWorks2016;
    SELECT LastName, FirstName
    FROM Person.Person AS c JOIN HumanResources.Employee AS e
    ON e.BusinessEntityID = c.BusinessEntityID
    WHERE 5000 IN (0.00);
    

    由于这是 false,因此 的行 Syed Abbas 不包含在具有相关子查询的上一个示例查询的结果中。 对 Pamela Ansman-Wolfe 行运行相同的过程。 你将看到此行包含在结果中,因为 WHERE 5000 IN (5000) 包括结果。

    通过在外部查询中引用表中的列作为表值函数的参数,相关子查询也可以在 FROM 子句中包含表值函数。 在这种情况下,对于外部查询的每一行,将根据子查询计算表值函数。

    子查询类型

    可以在许多位置指定子查询:

  • 使用别名。 有关详细信息,请参阅使用别名的子查询
  • 使用 INNOT IN。 有关详细信息,请参阅使用 IN 的子查询使用 NOT IN 的子查询
  • UPDATEDELETEINSERT 语句中。 有关详细信息,请参阅 UPDATE、DELETE 和 INSERT 语句中的子查询
  • 使用比较运算符。 有关详细信息,请参阅使用比较运算符的子查询
  • 使用 ANYSOMEALL。 有关详细信息,请参阅用 ANY、SOME 或 ALL 修改的比较运算符
  • 使用 IS [NOT] DISTINCT FROM。 有关详细信息,请参阅 IS [NOT] DISTINCT FROM (Transact-SQL)
  • 使用 EXISTSNOT EXISTS。 有关详细信息,请参阅使用 EXISTS 的子查询使用 NOT EXISTS 的子查询
  • 代替表达式。 有关详细信息,请参阅用于替代表达式的子查询
  • 带有表别名的子查询

    许多其中的子查询和外部查询引用同一表的语句可称为自联接(将某个表与自身联接)。 例如,您可以使用子查询查找特定州的员工的地址。

    USE AdventureWorks2016;
    SELECT StateProvinceID, AddressID
    FROM Person.Address
    WHERE AddressID IN
        (SELECT AddressID
         FROM Person.Address
         WHERE StateProvinceID = 39);
    

    下面是结果集。

    StateProvinceID AddressID
    ----------- -----------
    39 942
    39 955
    39 972
    39 22660
    (4 row(s) affected)
    

    也可以使用自联接:

    USE AdventureWorks2016;
    SELECT e1.StateProvinceID, e1.AddressID
    FROM Person.Address AS e1
    INNER JOIN Person.Address AS e2
    ON e1.AddressID = e2.AddressID
    AND e2.StateProvinceID = 39;
    

    表别名 e1e2 是必需的,因为要联接到自身的表以两个不同的角色出现。 别名也可用于在内部查询和外部查询中引用同一表的嵌套查询。

    USE AdventureWorks2016;
    SELECT e1.StateProvinceID, e1.AddressID
    FROM Person.Address AS e1
    WHERE e1.AddressID IN
        (SELECT e2.AddressID
         FROM Person.Address AS e2
         WHERE e2.StateProvinceID = 39);
    

    显式表别名表明,子查询中对 Person.Address 的引用与外部查询中的引用并不相同。

    使用 IN 的子查询

    通过 IN(或 NOT IN)引入的子查询结果是包含零个值或多个值的列表。 子查询返回结果之后,外部查询将利用这些结果。 下面的查询查找 Adventure Works Cycles 生成的所有车轮产品的名称。

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ProductSubcategoryID IN
        (SELECT ProductSubcategoryID
         FROM Production.ProductSubcategory
         WHERE [Name] = 'Wheels');
    

    下面是结果集:

    ---------------------------- LL Mountain Front Wheel ML Mountain Front Wheel HL Mountain Front Wheel LL Road Front Wheel ML Road Front Wheel HL Road Front Wheel Touring Front Wheel LL Mountain Rear Wheel ML Mountain Rear Wheel HL Mountain Rear Wheel LL Road Rear Wheel ML Road Rear Wheel HL Road Rear Wheel Touring Rear Wheel (14 row(s) affected)

    该语句分两步进行评估。 首先,内部查询返回与名称 'Wheel' (17) 匹配的子类别标识号。 其次,此值将替换为外部查询,该查询查找与 中的 Production.Product子类别标识号一起的产品名称。

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ProductSubcategoryID IN ('17');
    

    使用联接而不使用子查询处理该问题及类似问题的一个不同之处在于,联接使您可以在结果中显示多个表中的列。 例如,如果要在结果中包括产品子类别的名称,则必须使用联接版本。

    USE AdventureWorks2016;
    SELECT p.[Name], s.[Name]
    FROM Production.Product p
    INNER JOIN Production.ProductSubcategory s
    ON p.ProductSubcategoryID = s.ProductSubcategoryID
    AND s.[Name] = 'Wheels';
    

    下面是结果集。

    LL Mountain Front Wheel Wheels ML Mountain Front Wheel Wheels HL Mountain Front Wheel Wheels LL Road Front Wheel Wheels ML Road Front Wheel Wheels HL Road Front Wheel Wheels Touring Front Wheel Wheels LL Mountain Rear Wheel Wheels ML Mountain Rear Wheel Wheels HL Mountain Rear Wheel Wheels LL Road Rear Wheel Wheels ML Road Rear Wheel Wheels HL Road Rear Wheel Wheels Touring Rear Wheel Wheels (14 row(s) affected)

    以下示例查找所有信誉度良好,与 Adventure Works Cycles 有过至少 20 项订购记录,并且其平均交付时间小于 16 天的所有供应商的名称。

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Purchasing.Vendor
    WHERE CreditRating = 1
    AND BusinessEntityID IN
        (SELECT BusinessEntityID
         FROM Purchasing.ProductVendor
         WHERE MinOrderQty >= 20
         AND AverageLeadTime < 16);
    

    下面是结果集:

    -------------------------------------------------- Compete Enterprises, Inc International Trek Center First National Sport Co. Comfort Road Bicycles Circuit Cycles First Rate Bicycles Jeff's Sporting Goods Competition Bike Training Systems Electronic Bike Repair & Supplies Crowley Sport Expert Bike Co Team Athletic Co. Compete, Inc. (13 row(s) affected)

    评估内部查询后,产生符合子查询限定条件的供应商的 ID 号。 然后评估外部查询。 注意,在内部和外部查询的 WHERE 子句中,都可以包括多个条件。

    使用联接,同一查询可以用如下方式表示:

    USE AdventureWorks2016;
    SELECT DISTINCT [Name]
    FROM Purchasing.Vendor v
    INNER JOIN Purchasing.ProductVendor p
    ON v.BusinessEntityID = p.BusinessEntityID
    WHERE CreditRating = 1
      AND MinOrderQty >= 20
      AND AverageLeadTime < 16;
    

    联接总是可以表示为子查询。 子查询经常(但不总是)可以表示为联接。 这是因为联接是对称的:无论以何种顺序联接表 A 和 B,都将得到相同的结果。 如果涉及子查询,则情况并非如此。

    使用 NOT IN 的子查询

    通过 NOT IN 关键字引入的子查询也返回一列零值或更多值。 以下查询查找不是成品自行车的产品的名称。

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ProductSubcategoryID NOT IN
        (SELECT ProductSubcategoryID
         FROM Production.ProductSubcategory
         WHERE [Name] = 'Mountain Bikes'
            OR [Name] = 'Road Bikes'
            OR [Name] = 'Touring Bikes');
    

    此语句无法转换为联接。 类似的不相等联接具有不同的含义:它查找某个子类别中不是成品自行车的产品的名称。

    DELETEINSERT 语句中的UPDATE子查询

    可以在 UPDATEDELETEINSERTSELECT 数据操作 (DML) 语句中嵌套子查询。

    以下示例使 Production.Product 表的 ListPrice 列中的值加倍。 WHERE 子句中的子查询将引用 Purchasing.ProductVendor 表以便将 Product 表中更新的行仅限制为 BusinessEntity 1540 对应的那些行。

    USE AdventureWorks2016;
    UPDATE Production.Product
    SET ListPrice = ListPrice * 2
    WHERE ProductID IN
        (SELECT ProductID
         FROM Purchasing.ProductVendor
         WHERE BusinessEntityID = 1540);
    

    下面是一条使用联接的等效 UPDATE 语句:

    USE AdventureWorks2016;
    UPDATE Production.Product
    SET ListPrice = ListPrice * 2
    FROM Production.Product AS p
    INNER JOIN Purchasing.ProductVendor AS pv
        ON p.ProductID = pv.ProductID AND BusinessEntityID = 1540;
    

    为清楚起见,如果同一个表本身在其他子查询中引用,请使用目标表的别名:

    USE AdventureWorks2016;
    UPDATE p
    SET ListPrice = ListPrice * 2
    FROM Production.Product AS p
    INNER JOIN Purchasing.ProductVendor AS pv
        ON p.ProductID = pv.ProductID AND BusinessEntityID = 1540;
    

    使用比较运算符的子查询

    子查询可以由一个比较运算符(=< >>> =<! >! << =)引入。

    与使用 IN 引入的子查询一样,由未修饰的比较运算符(即后面不接 ANYALL 的比较运算符)引入的子查询必须返回单个值而不是值列表。 如果这样的子查询返回多个值,SQL Server 将显示一条错误信息。

    要使用由未修改的比较运算符引入的子查询,必须对数据和问题的本质非常熟悉,以了解该子查询实际是否只返回一个值。

    例如,如果假定每个销售员只负责一块销售区域,而你要找出 Linda Mitchell 所负责区域的客户,那么可以编写一条语句,带上由简单的 = 比较运算符引入的子查询。

    USE AdventureWorks2016;
    SELECT CustomerID
    FROM Sales.Customer
    WHERE TerritoryID =
        (SELECT TerritoryID
         FROM Sales.SalesPerson
         WHERE BusinessEntityID = 276);
    

    但是,如果 Linda Mitchell 负责的销售区域不止一个,则会产生一条错误信息。 这时可以用 IN 表达式(=ANY 也可以)来代替 = 比较运算符。

    由未修改的比较运算符引入的子查询经常包括聚合函数,因为这些子查询要返回单个值。 例如,下面的语句将找出定价高于平均定价的所有产品的名称。

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ListPrice >
        (SELECT AVG (ListPrice)
         FROM Production.Product);
    

    由于使用未修改的比较运算符引入的子查询必须返回单个值,因此除非你知道 GROUP BY 或 子句本身返回单个值,否则它们不能包含 GROUP BYHAVINGHAVING 子句。 例如,下面的查询将找出 ProductSubcategoryID 14 中定价高于最低定价的产品。

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ListPrice >
        (SELECT MIN (ListPrice)
         FROM Production.Product
         GROUP BY ProductSubcategoryID
         HAVING ProductSubcategoryID = 14);
    

    由 、 SOME或 修改的ANY比较运算符ALL

    可以用 ALLANY 关键字修改引入子查询的比较运算符。 SOME 是与 ANY 等效的 ISO 标准。 有关这些比较运算符的详细信息,请参阅 SOME | ANY

    通过修改的比较运算符引入的子查询返回零个值或多个值的列表,并且可以包括 GROUP BYHAVING 子句。 这些子查询可以用 EXISTS 重新表述。

    以 > 比较运算符为例, > ALL 表示大于每个值。 换句话说,它表示大于最大值。 例如,> ALL (1, 2, 3) 表示大于 3。 > ANY 表示至少大于一个值,即大于最小值。 因此 > ANY (1, 2, 3) 表示大于 1。

    若要使带有 > ALL 的子查询中的行满足外部查询中指定的条件,引入子查询的列中的值必须大于子查询返回的值列表中的每个值。

    同样,> ANY 表示要使某一行满足外部查询中指定的条件,引入子查询的列中的值必须至少大于子查询返回的值列表中的一个值。

    下面的查询提供一个由 ANY 修改的比较运算符引入的子查询的示例。 它查找定价高于或等于任何产品子类别的最高定价的产品。

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ListPrice >= ANY
        (SELECT MAX (ListPrice)
         FROM Production.Product
         GROUP BY ProductSubcategoryID);
    

    对于每个产品子类别,内部查询查找最高定价。 外部查询查看所有这些值,并确定定价高于或等于任何产品子类别的最高定价的单个产品。 如果 ANY 更改为 ALL,查询将只返回定价高于或等于内部查询返回的所有定价的那些产品。

    如果子查询未返回任何值,则整个查询无法返回任何值。

    = ANY 运算符等效于 IN。 例如,若要查找 Adventure Works Cycles 生产的所有轮子产品的名称,可以使用 IN= ANY

    --Using = ANY
    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ProductSubcategoryID = ANY
        (SELECT ProductSubcategoryID
         FROM Production.ProductSubcategory
         WHERE Name = 'Wheels');
    --Using IN
    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ProductSubcategoryID IN
        (SELECT ProductSubcategoryID
         FROM Production.ProductSubcategory
         WHERE Name = 'Wheels');
    

    下面是任一查询的结果集:

    -------------------------------------------------- LL Mountain Front Wheel ML Mountain Front Wheel HL Mountain Front Wheel LL Road Front Wheel ML Road Front Wheel HL Road Front Wheel Touring Front Wheel LL Mountain Rear Wheel ML Mountain Rear Wheel HL Mountain Rear Wheel LL Road Rear Wheel ML Road Rear Wheel HL Road Rear Wheel Touring Rear Wheel (14 row(s) affected)

    但是,<> ANY 运算符不同于 NOT IN

  • <> ANY 表示不等于 a,或者不等于 b,或者不等于 c
  • NOT IN 表示不等于 a,并且不等于 b,并且不等于 c
  • <> ALL 表示与 NOT IN 相同
  • 例如,以下查询查找位于任何销售人员都不负责的地区的客户。

    USE AdventureWorks2016;
    SELECT CustomerID
    FROM Sales.Customer
    WHERE TerritoryID <> ANY
        (SELECT TerritoryID
         FROM Sales.SalesPerson);
    

    结果包含除销售地区为 NULL 的客户以外的所有客户,因为分配给客户的每个地区都由一个销售人员负责。 内部查询查找销售人员涵盖的所有销售区域,然后,对于每个区域,外部查询查找不在一个区域的客户。

    由于同一原因,当在此查询中使用 NOT IN 时,结果将不包含任何客户。

    还可以使用 <> ALL 运算符获得相同的结果,该运算符与 NOT IN 等效。

    使用 EXISTS 的子查询

    使用 EXISTS 关键字引入子查询后,子查询的作用就相当于进行存在测试。 外部查询的 WHERE 子句测试子查询返回的行是否存在。 子查询实际上不会生成任何数据;它返回值 TRUEFALSE

    使用 EXISTS 引入的子查询的语法如下:

    WHERE [NOT] EXISTS (subquery)

    以下查询将查找 Wheels 子类别中所有产品的名称:

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE EXISTS
        (SELECT *
         FROM Production.ProductSubcategory
         WHERE ProductSubcategoryID =
                Production.Product.ProductSubcategoryID
            AND [Name] = 'Wheels');
    

    下面是结果集:

    -------------------------------------------------- LL Mountain Front Wheel ML Mountain Front Wheel HL Mountain Front Wheel LL Road Front Wheel ML Road Front Wheel HL Road Front Wheel Touring Front Wheel LL Mountain Rear Wheel ML Mountain Rear Wheel HL Mountain Rear Wheel LL Road Rear Wheel ML Road Rear Wheel HL Road Rear Wheel Touring Rear Wheel (14 row(s) affected)

    若要了解此查询的结果,请依次考虑每件产品的名称。 此值是否使子查询至少返回一行? 换句话说,查询是否使存在测试的计算结果为 TRUE

    注意,使用 EXISTS 引入的子查询在下列方面与其他子查询略有不同:

  • 关键字 (keyword) EXISTS前面没有列名、常量或其他表达式。
  • EXISTS 引入的子查询的选择列表通常几乎都是由星号 (*) 组成。 没有理由列出列名,因为你只是测试是否存在满足子查询中指定的条件的行。
  • 由于通常没有备选的、无子查询的表示法,因此 EXISTS 关键字很重要。 虽然使用 EXISTS 创建的一些查询无法以任何其他方式表示,但许多查询可以使用 IN 或 由 ANYALL 修改的比较运算符来实现类似的结果。

    例如,可以使用 IN 表示上述查询:

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE ProductSubcategoryID IN
        (SELECT ProductSubcategoryID
         FROM Production.ProductSubcategory
         WHERE [Name] = 'Wheels');
    

    使用 NOT EXISTS 的子查询

    NOT EXISTSEXISTS 的工作方式类似,只是如果子查询不返回行,那么使用 NOT EXISTS 的 WHERE 子句会得到令人满意的结果。

    例如,若要查找不在 wheel 子类别中的产品名称,请执行以下操作:

    USE AdventureWorks2016;
    SELECT [Name]
    FROM Production.Product
    WHERE NOT EXISTS
        (SELECT *
         FROM Production.ProductSubcategory
         WHERE ProductSubcategoryID =
                Production.Product.ProductSubcategoryID
            AND [Name] = 'Wheels');
    

    用于替代表达式的子查询

    在 Transact-SQL 中,可以在 、、 和 语句中使用SELECTUPDATE表达式的任何位置替换子查询,但列表中除外ORDER BYDELETEINSERT

    以下示例说明如何使用此增强功能。 此查询找出所有山地车产品的价格、平均价格以及两者之间的差价。

    USE AdventureWorks2016;
    SELECT [Name], ListPrice,
    (SELECT AVG(ListPrice) FROM Production.Product) AS Average,
        ListPrice - (SELECT AVG(ListPrice) FROM Production.Product)
        AS Difference
    FROM Production.Product
    WHERE ProductSubcategoryID = 1;
    
  • IN (Transact-SQL)
  • EXISTS (Transact-SQL)
  • ALL (Transact-SQL)
  • SOME | ANY (Transact-SQL)
  • IS [NOT] DISTINCT FROM (Transact-SQL)
  • 比较运算符 (Transact-SQL)
  • 查询性能概念

  • 查询处理体系结构指南
  • 查询存储最佳做法
  • SQL 数据库中的智能查询处理
  • 基数估计 (SQL Server)
  •