SQL148 返回产品名称和每一项产品的总订单数

描述

Products表为产品信息表含有字段prod_id产品id、prod_name产品名称

prod_idprod_name
a0001egg
a0002sockets
a0013coffee
a0003cola
a0023soda

OrderItems表为订单信息表含有字段order_num订单号和产品id prod_id

prod_idorder_num
a0001a105
a0002a1100
a0002a200
a0013a1121
a0003a10
a0003a19
a0003a5

【问题】

使用 OUTER JOIN 联结 Products 表和 OrderItems 表,返回产品名称(prod_name)和每一项产品的总订单数(不是订单号),并按产品名称升序排序。

【示例结果】

返回产品名称prod_name和订单号订单数orders

prod_nameorders
coffee1
cola3
egg1
sockets2
soda0

【示例解析】

返回产品和产品对应的实际支付的订单数,但是无实际订单的产品soda也返回,最后根据产品名称升序排序。

SELECT P.prod_name, IFNULL(T.orders, 0) AS orders 
FROM Products P 
LEFT JOIN (
    SELECT prod_id, COUNT(*) AS orders 
    FROM OrderItems 
    GROUP BY prod_id
) T ON T.prod_id = P.prod_id
ORDER BY P.prod_name;

 关键代码解释

1. 处理 NULLIFNULL(T.orders, 0)

在使用 LEFT JOIN 时,如果右表(这里是子查询 T)中没有匹配的记录,对应字段(如 T.orders)的值就会是 NULL
但在业务展示中,我们通常希望“没有订单”显示为 0,而不是 NULL 或空值。

IFNULL()MySQL 特有的函数,用于判断并替换 NULL 值:

IFNULL(expression, replacement)
  • 如果 expression 不是 NULL,返回该表达式的值;
  • 如果 expression 是 NULL,则返回 replacement

✅ 应用示例:

IFNULL(T.orders, 0)

T.ordersNULL(即产品从未被下单)时,返回 0,实现“零订单显示为 0”的需求。

⚠️ 注意IFNULL 仅适用于 MySQL 数据库。在其他数据库系统(如 PostgreSQL、SQL Server、Oracle)中不支持。


2. 更通用的替代方案:COALESCE(T.orders, 0)

如果你使用的是其他数据库,或希望写出兼容性更强的标准 SQL,应使用 COALESCE() 函数。

COALESCE(value1, value2, ..., valueN)
  • 返回参数列表中第一个非 NULL 的值
  • 至少需要两个参数。
  • 是 SQL 标准函数,几乎在所有关系型数据库中都支持(MySQL、PostgreSQL、SQL Server、Oracle、SQLite 等)。

✅ 应用示例:

COALESCE(T.orders, 0)

含义与 IFNULL(T.orders, 0) 完全相同:如果 T.ordersNULL,就返回 0

优势

  • 跨数据库兼容性好
  • 功能更强大(可接受多个参数,例如 COALESCE(col1, col2, 0) 表示优先取 col1,若为 NULL 则取 col2,都为 NULL 才返回 0

推荐使用场景

  • 你不确定数据库类型
  • 项目要求使用标准 SQL
  • 需要更灵活的 NULL 处理逻辑

✅ 总结对比

函数语法数据库支持是否标准 SQL推荐场景
IFNULLIFNULL(expr, default)仅 MySQL❌ 否MySQL 项目,追求简洁
COALESCECOALESCE(expr1, expr2, ...)所有主流数据库✅ 是通用项目、跨数据库、标准 SQL

易错点 

📝 为什么统计订单要“先聚合,再连接”?

❌ 错误做法:直接连接

SELECT P.prod_name, COUNT(*) 
FROM Products P 
LEFT JOIN OrderItems O ON P.prod_id = O.prod_id
GROUP BY P.prod_name;

问题:

  • 逻辑混乱:COUNT(*) 数的是连接后的行数,看似正确,实则侥幸

  • 性能差:先产生大量重复数据,再分组统计。

  • 难扩展:一旦要加条件或字段,容易出错。


✅ 正确做法:先聚合,再连接

SELECT P.prod_name, COALESCE(T.orders, 0) AS orders
FROM Products P
LEFT JOIN (
    SELECT prod_id, COUNT(*) AS orders 
    FROM OrderItems 
    GROUP BY prod_id
) T ON P.prod_id = T.prod_id;

优势:

优点

说明

✅ 逻辑清晰

子查询专注“统计”,主查询专注“拼接”

✅ 性能更好

聚合在小表上完成,连接更高效

✅ 避免重复

不会因多订单产生重复行

✅ 易维护

修改统计条件只需动子查询


💡 类比理解

  • OrderItems 是几千张小票。

  • 你想知道“每个产品卖了多少”。

❌ 别把所有小票和产品表连起来数!
✅ 先让系统出个《销量统计表》,再贴到产品表旁边。

这个“统计表”就是子查询。


✅ 核心原则

“聚合在前,连接在后”
先用子查询生成统计结果,再用 LEFT JOIN 关联主表,
配合 COALESCE 处理 NULL,确保未销售产品显示为 0

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值