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

目标:

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;

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,193评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,306评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,130评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,110评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,118评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,085评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,007评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,844评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,283评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,508评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,395评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,985评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,630评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,797评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,653评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,553评论 2 352

推荐阅读更多精彩内容