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

相关推荐
韩立学长8 分钟前
基于Springboot的旧时月历史论坛4099k6s9(程序、源码、数据库、调试部署方案及开发环境)系统界面展示及获取方式置于文档末尾,可供参考。
数据库·spring boot·后端
TDengine (老段)1 小时前
TDengine 字符串函数 CONCAT_WS 用户手册
android·大数据·数据库·时序数据库·tdengine·涛思数据
IT 小阿姨(数据库)2 小时前
PostgreSQL 之上的开源时序数据库 TimescaleDB 详解
运维·数据库·sql·postgresql·开源·centos·时序数据库
熊文豪3 小时前
openEuler 云原生实战:部署高性能 Redis 集群与压测分析
数据库·redis·云原生·openeuler
GTgiantech3 小时前
科普SFP 封装光模块教程
服务器·网络·数据库
深圳市恒讯科技3 小时前
如何在服务器上安装和配置数据库(如MySQL)?
服务器·数据库·mysql
言之。3 小时前
TiDB分布式数据库技术架构概述
数据库·分布式·tidb
万事大吉CC3 小时前
SQL表设计与约束教程
数据库·sql
员大头硬花生4 小时前
七、InnoDB引擎-架构-后台线程
java·数据库·mysql
Ryan ZX4 小时前
etcd 高可用分布式键值存储
数据库·分布式·etcd