美文网首页
《SQL必知必会》第 11 课 使用子查询

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

作者: 一条快乐的鱼儿 | 来源:发表于2021-12-01 15:53 被阅读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;

相关文章

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

    目标: 11.1 子查询 11.2 利用子查询进行过滤 11.3 作为计算字段使用子查询 11.1 子查询 查询(...

  • SQL必知必会

    《SQL必知必会》SQL是使用 广泛的数据库语言,几乎所有重要的DBMS都支持SQL。《SQL必知必会(第4版)》...

  • SQL必知必会

    SQL 必知必会 第1章 了解SQL 定义 结构化查询语言(Structured Query Language) ...

  • SQL必知必会(子查询)

    一、什么是关联子查询,什么是非关联子查询 子查询虽然是一种嵌套查询的形式,不过我们依然可以依据子查询是否执行多次,...

  • SQL 学习笔记

    SQL基础 参考书籍: 《SQL必知必会》 检索数据 过滤数据 函数 分组数据 子查询 简单联结 法一: 法二: ...

  • 第三周学习计划

    本周将继续学习SQL必知必会这本书,从第8课开始。

  • 《SQL必知必会 第4版》PDF高清完整版-免费下载

    《SQL必知必会 第4版》PDF高清完整版-免费下载 《SQL必知必会 第4版》PDF高清完整版-免费下载 下载地...

  • 初学数据分析

    本周学习SQL。 SQL必知必会第一课第二课部分

  • 第三周学习总结

    本周继续学习了SQL必知必会这本书,从第8课开始,已学到第21课使用游标。本周主要学习了用函数处理汇总数据,分组查...

  • mysql必知必会

    title: mysql必知必会date: 2019-11-21tags: 笔记categories: sql ...

网友评论

      本文标题:《SQL必知必会》第 11 课 使用子查询

      本文链接:https://www.haomeiwen.com/subject/ifopaltx.html