183. 从不订购的客户
Customers 表:
±------------±--------+
| Column Name | Type |
±------------±--------+
| id | int |
| name | varchar |
±------------±--------+
在 SQL 中,id 是该表的主键。
该表的每一行都表示客户的 ID 和名称。
Orders 表:
±------------±-----+
| Column Name | Type |
±------------±-----+
| id | int |
| customerId | int |
±------------±-----+
在 SQL 中,id 是该表的主键。
customerId 是 Customers 表中 ID 的外键( Pandas 中的连接键)。
该表的每一行都表示订单的 ID 和订购该订单的客户的 ID。
找出所有从不点任何东西的顾客。
以 任意顺序 返回结果表。
结果格式如下所示。
示例 1:
输入:
Customers table:
±—±------+
| id | name |
±—±------+
| 1 | Joe |
| 2 | Henry |
| 3 | Sam |
| 4 | Max |
±—±------+
Orders table:
±—±-----------+
| id | customerId |
±—±-----------+
| 1 | 3 |
| 2 | 1 |
±—±-----------+
输出:
±----------+
| Customers |
±----------+
| Henry |
| Max |
±----------+
题解
- 该表的每一行都表示订单的 ID 和订购该订单的客户的 ID。
啥子含义?就是明细表呗,主键唯一,客户唯一,订单唯一,客户与订单是1对n的关系,也就是说订单是最完整的数据,客户可能没有订单,但有了订单已经能找到对应的客户。 - 找出所有从不点任何东西的顾客。
从来不点任何东西,啥子含义?说白了就是在订单明细表中不存在任何一条记录的客户。
那这样是不是就分析清晰了?怎么判断这个表里的数据有没有在另外一个表里呢? join拉齐? union很显然不合适
还要判断不在某一个表中的数据呢?外连接比较合适,关联不上会置空嘛
于是乎方法一诞生
方法一 外连接+where
select
c1.name as customers
from customers c1 left join orders o1 on c1.id=o1.customerId
where o1.id is null
方法二 子查询
什么思路呢?先找到点单的有哪些,再逆向思维取反,搞定
select
name as customers
from customers
where id not in(select distinct customerId from orders)
方法三 不用in还有什么吗?not exists
思路同上,找到另外一张表不存在的记录,但exists 会比子查询性能更佳
not exists (select distinct customerId from orders o where c.id=o.customerId)
这是整体,exists 返回值是 true或者false,可能会有人看不懂,简单解释下
select
name as customers
from customers c
where not exists (select distinct customerId from orders o where c.id=o.customerId)
这里贴个图就不多bb拉