第11课 使用子查询

第11课 使用子查询
11.1 子查询
子查询(subquery):即嵌套在其他查询中的查询。
 
11.2 利用子查询进行过滤
订单存储在两个表中。每个订单包含订单编号、客户 ID、订单日期,在 Orders 表中存储为一行。各订单的物品存储在相关的OrderItems 表中。Orders 表不存储顾客信息,只存储顾客 ID。顾客的
实际信息存储在 Customers 表中。
现在,假如需要列出订购物品 RGAN01 的所有顾客,应该怎样检索?下面列出具体的步骤。
(1) 检索包含物品 RGAN01 的所有订单的编号。
(2) 检索具有前一步骤列出的订单编号的所有顾客的 ID。
(3) 检索前一步骤返回的所有顾客 ID 的顾客信息。
上述每个步骤都可以单独作为一个查询来执行。可以把一条 SELECT 语句返回的结果用于另一条 SELECT 语句的 WHERE 子句。
也可以使用子查询来把 3 个查询组合成一条语句。
 
第一条 SELECT 语句的含义很明确,它对 prod_id 为 RGAN01 的所有订单物品,检索其 order_num 列。输出列出了两个包含此物品的订单:
SELECT order_num
FROM OrderItems
WHERE prod_id = 'RGAN01';
 
输出结果如下:
order_num
-----------
20007
20008
 
现在,我们知道了哪个订单包含要检索的物品,下一步查询与订单 20007和 20008 相关的顾客 ID。利用第 5 课介绍的 IN 子句,编写如下的 SELECT语句:
SELECT cust_id
FROM Orders
WHERE order_num IN (20007,20008);
 
输出结果如下:
cust_id
----------
1000000004
1000000005
 
现在,结合这两个查询,把第一个查询(返回订单号的那一个)变为子
查询。请看下面的 SELECT 语句:
SELECT cust_id
FROM Orders
WHERE order_num IN (SELECT order_num
FROM OrderItems
WHERE prod_id = 'RGAN01');
 
输出结果如下:
cust_id
----------
1000000004
1000000005
 
分析:
首先,它执行下面的查询:
SELECT order_num FROM orderitems WHERE prod_id='RGAN01'
此查询返回两个订单号:20007 和 20008。然后,这两个值以 IN 操作符要求的逗号分隔的格式传递给外部查询的 WHERE 子句。外部查询变成:
SELECT cust_id FROM orders WHERE order_num IN (20007,20008)
可以看到,输出是正确的,与前面硬编码 WHERE 子句所返回的值相同。
 
现在得到了订购物品 RGAN01 的所有顾客的 ID。下一步是检索这些顾客ID 的顾客信息。检索两列的 SQL 语句为:
SELECT cust_name, cust_contact
FROM Customers
WHERE cust_id IN ('1000000004','1000000005');
 
可以把其中的 WHERE 子句转换为子查询,而不是硬编码这些顾客 ID:
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'));
 
输出结果如下:
cust_name cust_contact
----------------------------- --------------------
Fun4All Denise L. Stephens
The Toy Store Kim Howard
 
分析:
为了执行上述 SELECT 语句,DBMS 实际上必须执行三条 SELECT 语句。最里边的子查询返回订单号列表,此列表用于其外面的子查询的 WHERE子句。外面的子查询返回顾客 ID 列表,此顾客 ID 列表用于最外层查询的 WHERE 子句。最外层查询返回所需的数据。
可见,在 WHERE 子句中使用子查询能够编写出功能很强且很灵活的 SQL语句。对于能嵌套的子查询的数目没有限制,不过在实际使用时由于性能的限制,不能嵌套太多的子查询。
 
11.3 作为计算字段使用子查询
 
posted @ 2022-06-22 07:27  lqsj2018  阅读(54)  评论(0编辑  收藏  举报