WITH在MYSQL中的用法

WITH 子句(也称为公共表表达式 ,Common Table Expression,简称 CTE)是 SQL 中一种强大的查询构建工具,它可以显著提高复杂查询的可读性和可维护性

一、基本语法结构

sql 复制代码
WITH cte_name AS (
    SELECT ...  -- 定义CTE的查询
)
SELECT ... FROM cte_name;  -- 主查询使用CTE

二、CTE 的核心特点

  1. 临时结果集:CTE 只在当前查询执行期间存在
  2. 可引用性:定义后可在主查询中多次引用
  3. 作用域限制:仅在紧随其后的单个语句中有效

三、MySQL 中 CTE 的具体用法

1. 基本 CTE(单表表达式)

sql 复制代码
WITH sales_summary AS (
    SELECT product_id, SUM(quantity) AS total_sold
    FROM orders
    GROUP BY product_id
)
SELECT p.product_name, s.total_sold
FROM products p
JOIN sales_summary s ON p.product_id = s.product_id;

2. 多 CTE 定义(逗号分隔)

sql 复制代码
WITH 
customer_orders AS (
    SELECT customer_id, COUNT(*) AS order_count
    FROM orders
    GROUP BY customer_id
),
high_value_customers AS (
    SELECT customer_id
    FROM customer_orders
    WHERE order_count > 5
)
SELECT c.customer_name
FROM customers c
JOIN high_value_customers h ON c.customer_id = h.customer_id;

3. 递归 CTE(MySQL 8.0+ 支持)

递归 CTE 用于处理层次结构数据:

sql 复制代码
WITH RECURSIVE org_hierarchy AS (
    -- 基础查询(锚成员)
    SELECT id, name, parent_id, 1 AS level
    FROM organization
    WHERE parent_id IS NULL
    
    UNION ALL
    
    -- 递归查询(递归成员)
    SELECT o.id, o.name, o.parent_id, h.level + 1
    FROM organization o
    JOIN org_hierarchy h ON o.parent_id = h.id
)
SELECT * FROM org_hierarchy;

四、CTE 的优势

  1. 提高可读性

    • 将复杂查询分解为逻辑块
    • 类似编程中的变量定义
  2. 避免重复子查询

    sql 复制代码
    -- 不使用CTE(重复子查询)
    SELECT * FROM (SELECT ... FROM table1) AS t1
    JOIN (SELECT ... FROM table1) AS t2...
    
    -- 使用CTE(避免重复)
    WITH t1 AS (SELECT ... FROM table1)
    SELECT * FROM t1 JOIN t1 AS t2...
  3. 支持递归查询:处理树形/层次结构数据

五、CTE 与临时表的区别

特性 CTE 临时表
生命周期 仅当前语句有效 会话结束前有效
存储 不物理存储 可能存储在内存或磁盘
索引 不能创建索引 可以创建索引
可见性 仅定义它的查询可见 同一会话的后续查询可见
性能 优化器可能内联展开 需要实际创建和填充

六、实际应用场景

1. 复杂报表查询

sql 复制代码
WITH 
monthly_sales AS (...),
product_ranking AS (...)
SELECT ... FROM monthly_sales JOIN product_ranking...

2. 数据清洗管道

sql 复制代码
WITH 
raw_data AS (...),
cleaned_data AS (...),
enriched_data AS (...)
SELECT * FROM enriched_data;

3. 层次结构遍历(组织架构、评论线程等)

sql 复制代码
WITH RECURSIVE comment_tree AS (...)
SELECT * FROM comment_tree;

七、性能注意事项

  1. 物化提示

    sql 复制代码
    WITH cte_name AS (
        SELECT /*+ MATERIALIZE */ ...  -- 强制物化
    )
  2. 合并提示

    sql 复制代码
    WITH cte_name AS (
        SELECT /*+ MERGE */ ...  -- 强制合并到主查询
    )
  3. 递归深度控制(MySQL 默认 1000):

    sql 复制代码
    SET @@cte_max_recursion_depth = 2000;

八、版本兼容性

  • MySQL 8.0+ 完整支持 CTE 和递归 CTE
  • MySQL 5.7 及更早版本不支持 CTE

WITH 子句是现代 SQL 开发中不可或缺的工具,合理使用可以大幅提升查询的清晰度和维护性,特别是在处理多层嵌套或递归数据时。

相关推荐
Li zlun19 分钟前
MySQL 性能监控与安全管理完全指南
数据库·mysql·安全
养生技术人1 小时前
Oracle OCP认证考试题目详解082系列第48题
运维·数据库·sql·oracle·database·开闭原则·ocp
海阳宜家电脑1 小时前
Lazarus使用TSQLQuery更新的一点技巧
数据库·lazarus·tsqlquery
丨我是张先生丨2 小时前
SQLSERVER 查找存储过程中某个变量
数据库
感谢地心引力2 小时前
【Python】基于 PyQt6 和 Conda 的 PyInstaller 打包工具
数据库·python·conda·pyqt·pyinstaller
韩立学长4 小时前
【开题答辩实录分享】以《走失人口系统档案的设计与实现》为例进行答辩实录分享
mysql·mybatis·springboot
lypzcgf4 小时前
Coze源码分析-资源库-编辑数据库-后端源码-数据存储层
数据库·coze·coze源码分析·智能体平台·ai应用平台
jackaroo20204 小时前
后端_Redis 分布式锁实现指南
数据库·redis·分布式
liuy96154 小时前
迷你论坛项目
数据库
杨云龙UP4 小时前
小工具大体验:rlwrap加持下的Oracle/MySQL/SQL Server命令行交互
运维·服务器·数据库·sql·mysql·oracle·sqlserver