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

描述

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

|---------|-----------|
| prod_id | prod_name |
| a0001 | egg |
| a0002 | sockets |
| a0013 | coffee |
| a0003 | cola |
| a0023 | soda |

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

|---------|-----------|
| prod_id | order_num |
| a0001 | a105 |
| a0002 | a1100 |
| a0002 | a200 |
| a0013 | a1121 |
| a0003 | a10 |
| a0003 | a19 |
| a0003 | a5 |

【问题】

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

【示例结果】

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

|-----------|--------|
| prod_name | orders |
| coffee | 1 |
| cola | 3 |
| egg | 1 |
| sockets | 2 |
| soda | 0 |

【示例解析】

返回产品和产品对应的实际支付的订单数,但是无实际订单的产品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,返回该表达式的值;
  • 如果 expressionNULL,则返回 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 推荐场景
IFNULL IFNULL(expr, default) 仅 MySQL ❌ 否 MySQL 项目,追求简洁
COALESCE COALESCE(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

相关推荐
云动雨颤5 小时前
Spring Boot配置优化:Tomcat+数据库+缓存+日志,全场景教程
数据库·spring boot·tomcat
RestCloud5 小时前
Kafka实时数据管道:ETL在流式处理中的应用
数据库·kafka·api
寻星探路6 小时前
数据库造神计划第九天---增删改查(CRUD)(5)
数据库
Alan521597 小时前
🚀 阿里云 ECS + MySQL 环境搭建全流程(用于个人博客系统开发)
数据库·程序员
Huhbbjs7 小时前
SQL 核心概念与实践总结
开发语言·数据库·sql
wuyunhang1234567 小时前
Redis---集群模式
数据库·redis·缓存
sensenlin917 小时前
Mybatis中SQL全大写或全小写影响执行性能吗
数据库·sql·mybatis
IAtlantiscsdn8 小时前
Redis Stack扩展功能
java·数据库·redis
没有bug.的程序员9 小时前
Redis 大 Key 与热 Key:生产环境的风险与解决方案
java·数据库·redis·缓存·热key·大key