《SQL必知必会》第 11 课 使用子查询

2021-12-01  本文已影响0人  一条快乐的鱼儿

目标:

11.1 子查询

11.2 利用子查询进行过滤

11.3 作为计算字段使用子查询


11.1 子查询

查询(query)任何 SQL 语句都是查询。但此术语一般指 SELECT 语句。

SQL 还允许创建子查询(subquery),即嵌套在其他查询中的查询。

11.2 利用子查询进行过滤

要求:假如需要列出订购物品 RGAN01 的所有顾客,应该怎样检索?下

面列出具体的步骤。

(1) 检索包含物品 RGAN01 的所有订单的编号。

(2) 检索具有前一步骤列出的订单编号的所有顾客的 ID。

(3) 检索前一步骤返回的所有顾客 ID 的顾客信息。

SELECT order_num

FROM OrderItems

WHERE prod_id = 'RGAN01';

SELECT cust_id

FROM Orders

WHERE order_num IN (20007,20008);

#嵌套

SELECT cust_id

FROM Orders

WHERE order_num IN (SELECT order_num

                                        FROM OrderItems

                                        WHERE prod_id = 'RGAN01');

输出:# cust_id'1000000004''1000000005'

分析:在 SELECT 语句中,子查询总是从内向外处理。

SELECT cust_name, cust_contact

FROM Customers

WHERE cust_id IN (SELECT cust_id

                                 FROM Orders

                                WHERE order_num IN (SELECT order_num

                                                                       FROM OrderItems

                                                                       WHERE prod_id = 'RGAN01'));

三条SELECT语句

注意:只能是单列

作为子查询的 SELECT 语句只能查询单个列。企图检索多个列将返回错误。

注意:子查询和性能

这里给出的代码有效,并且获得了所需的结果。但是,使用子查询并不总是执行这类数据检索的最有效方法。

11.3 作为计算字段使用子查询

使用子查询的另一方法是创建计算字段。假如需要显示 Customers 表中每个顾客的订单总数。订单与相应的顾客 ID 存储在 Orders 表中。

执行这个操作,要遵循下面的步骤:

(1) 从 Customers 表中检索顾客列表;

(2) 对于检索出的每个顾客,统计其在 Orders 表中的订单数目。

SELECT cust_name, cust_state,

              (SELECT COUNT(*)

               FROM Orders

               WHERE Orders.cust_id = Customers.cust_id) AS orders

FROM Customers

ORDER BY cust_name;

子查询中的 WHERE 子句与前面使用的 WHERE 子句稍有不同,因为它使用了完全限定列名,而不只是列名(cust_id)。它指定表名和列名(Orders.cust_id和 Customers.cust_id).

注意:完全限定列名 你已经看到了为什么要使用完全限定列名,没有具体指定就会返回错误结果,因为 DBMS 会误解你的意思。有时候,由于出现冲突列名而导致的歧义性,会引起 DBMS 抛出错误信息。例如,WHERE 或 ORDER BY 子句指定的某个列名可能会出现在多个表中。好的做法是,如果在SELECT 语句中操作多个表,就应使用完全限定列名来避免歧义。

11.4小结

这一课学习了什么是子查询,如何使用它们。子查询常用于 WHERE 子句的 IN 操作符中,以及用来填充计算列。我们举了这两种操作类型的例子。

-- 11.2 利用子查询进行过滤

SELECT order_num

FROM OrderItems

WHERE prod_id = 'RGAN01';

SELECT cust_id

FROM Orders

WHERE order_num IN (20007,20008);

SELECT cust_id

FROM Orders

WHERE order_num IN (SELECT order_num

FROM OrderItems

WHERE prod_id = 'RGAN01');

SELECT cust_name, cust_contact

FROM Customers

WHERE cust_id IN (SELECT cust_id

                  FROM Orders

  WHERE order_num IN (SELECT order_num

  FROM OrderItems

  WHERE prod_id = 'RGAN01'));

-- 11.3 作为计算字段使用子查询

SELECT cust_name, cust_state,

      (SELECT COUNT(*)

        FROM Orders

        WHERE Orders.cust_id = Customers.cust_id) AS orders

FROM Customers

ORDER BY cust_name;

11.5挑战题

1. 使用子查询,返回购买价格为 10 美元或以上产品的顾客列表。你需要使用 OrderItems 表查找匹配的订单号(order_num),然后使用Order 表检索这些匹配订单的顾客 ID(cust_id)。

SELECT cust_id,cust_name

FROM orders

WHERE order_num IN(SELECT order_num

                                  FROM orderitems

                                 WHERE item_price>=10 );

2. 你想知道订购 BR01 产品的日期。编写 SQL 语句,使用子查询来确定哪些订单(在 OrderItems 中)购买了 prod_id 为 BR01 的产品,然后从 Orders 表中返回每个产品对应的顾客 ID(cust_id)和订单日期(order_date)。按订购日期对结果进行排序。

SELECT cust_id,order_date

FROM Orders

WHERE order_num IN (SELECT order_num

                    FROM OrderItems

                    WHERE prod_id='BR01')

order by order_date;

3. 现在我们让它更具挑战性。在上一个挑战题,返回购买 prod_id 为BR01 的产品的所有顾客的电子邮件(Customers 表中的 cust_email)。提示:这涉及 SELECT 语句,最内层的从 OrderItems 表返回 order_num,中间的从 Customers 表返回 cust_id。

SELECT cust_email,cust_name

FROM customers

WHERE cust_id IN(SELECT cust_id

    FROM Orders

                WHERE order_num IN (SELECT order_num

                                    FROM OrderItems

                                    WHERE prod_id='BR01'));

4. 我们需要一个顾客 ID 列表,其中包含他们已订购的总金额。编写 SQL语句,返回顾客 ID(Orders 表中的 cust_id),并使用子查询返回total_ordered 以便返回每个顾客的订单总数。将结果按金额从大到小排序。提示:你之前已经使用 SUM()计算订单总数。

SELECT cust_id,

      (SELECT SUM(item_price*quantity)

        FROM OrderItems

        WHERE Orders.order_num = OrderItems.order_num) AS total_ordered

FROM Orders

ORDER BY total_ordered DESC;

5. 再来。编写 SQL 语句,从 Products 表中检索所有的产品名称(prod_name),以及名为 quant_sold 的计算列,其中包含所售产品的总数(在 OrderItems 表上使用子查询和 SUM(quantity)检索)。

SELECT prod_name,

      (SELECT Sum(quantity)

        FROM OrderItems

        WHERE Products.prod_id=OrderItems.prod_id) AS quant_sold

FROM Products;

上一篇 下一篇

猜你喜欢

热点阅读