公用表表达式(CTE)详解:针对 MySQL 和 SQL Server 数据库

公用表表达式(CTE,Common Table Expressions)是一种在 SQL 中定义临时结果集的方法,该结果集在单个查询的执行过程中可以被引用。CTE 提高了查询的可读性和结构化,特别适用于复杂的子查询和递归查询。本文将详细介绍 CTE 的概念和用法,并分别针对 MySQL 和 SQL Server 数据库进行说明。

什么是公用表表达式(CTE)

CTE 是一个命名的临时结果集,它在查询的执行范围内有效。CTE 的定义在 WITH 子句中,紧随其后的是查询主体。CTE 分为两种类型:

  1. 非递归 CTE:用于定义一次性计算结果的临时表。
  2. 递归 CTE:用于处理递归查询,如处理层次结构数据(例如组织结构图)。

CTE 的语法

基本语法
sql 复制代码
WITH cte_name AS (
    SELECT column1, column2, ...
    FROM table_name
    WHERE condition
)
SELECT column1, column2, ...
FROM cte_name
WHERE condition;

MySQL 中的 CTE

MySQL 从 8.0 版本开始支持 CTE。下面是一些常见的使用场景。

示例 1:非递归 CTE

假设我们有一个名为 bs_bill_day 的表,并且希望使用 CTE 从该表中选择数据,然后限制结果为前 10 行。

sql 复制代码
WITH bill_cte AS (
    SELECT * FROM bs_bill_day
)
SELECT * 
FROM bill_cte 
LIMIT 10;
示例 2:递归 CTE

假设我们有一个表示员工和经理关系的表 employees,其中包含 employee_idmanager_id 列,我们希望找到某个经理及其所有下属。

sql 复制代码
WITH RECURSIVE employee_cte AS (
    -- 初始查询,选择顶层经理
    SELECT employee_id, manager_id, 1 AS level
    FROM employees
    WHERE manager_id IS NULL

    UNION ALL

    -- 递归部分,选择下一级员工
    SELECT e.employee_id, e.manager_id, ec.level + 1
    FROM employees e
    INNER JOIN employee_cte ec ON e.manager_id = ec.employee_id
)
SELECT * 
FROM employee_cte;

SQL Server 中的 CTE

SQL Server 从 2005 版本开始支持 CTE。它的语法和 MySQL 类似。

示例 1:非递归 CTE

假设我们有一个名为 bs_bill_day 的表,并且希望使用 CTE 从该表中选择数据,然后限制结果为前 10 行。

sql 复制代码
WITH bill_cte AS (
    SELECT * FROM bs_bill_day
)
SELECT TOP 10 * 
FROM bill_cte;
示例 2:递归 CTE

假设我们有一个表示员工和经理关系的表 employees,其中包含 employee_idmanager_id 列,我们希望找到某个经理及其所有下属。

sql 复制代码
WITH employee_cte AS (
    -- 初始查询,选择顶层经理
    SELECT employee_id, manager_id, 1 AS level
    FROM employees
    WHERE manager_id IS NULL

    UNION ALL

    -- 递归部分,选择下一级员工
    SELECT e.employee_id, e.manager_id, ec.level + 1
    FROM employees e
    INNER JOIN employee_cte ec ON e.manager_id = ec.employee_id
)
SELECT * 
FROM employee_cte;

使用 CTE 进行分页

在大数据集的分页查询中,CTE 也非常有用。以下是在 MySQL 和 SQL Server 中使用 CTE 进行分页的示例。

MySQL 分页
sql 复制代码
WITH bill_cte AS (
    SELECT * FROM bs_bill_day
)
SELECT * 
FROM bill_cte 
LIMIT 10 OFFSET 0; -- 获取第一页的10条记录

SQL Server 分页

sql 复制代码
WITH bill_cte AS (
    SELECT *, ROW_NUMBER() OVER (ORDER BY some_column) AS row_num
    FROM bs_bill_day
)
SELECT * 
FROM bill_cte
WHERE row_num BETWEEN 1 AND 10; -- 获取第一页的10条记录

mysql实现返回最近三十天的列

sql 复制代码
WITH RECURSIVE recent_day_cte AS (
    -- 初始查询,获取当前日期的订单记录
    SELECT CURDATE() AS dayKey

    UNION ALL

    -- 递归查询,获取前一天的订单记录
    SELECT DATE_SUB(dayKey, INTERVAL 1 DAY)
    FROM recent_day_cte
    WHERE dayKey >= CURDATE() - INTERVAL 29 DAY
)
SELECT *
FROM recent_day_cte;

sql server 实现返回最近三十天的列

sql 复制代码
WITH recent_dates_cte AS (
    -- 初始查询,获取当前日期的订单记录
    SELECT CAST(GETDATE() AS DATE) AS dayKey

    UNION ALL

    -- 递归查询,获取前一天的日期
    SELECT DATEADD(DAY, -1, dayKey)
    FROM recent_dates_cte
    WHERE dayKey >= DATEADD(DAY, -29, CAST(GETDATE() AS DATE))
)
SELECT *
FROM recent_dates_cte;

sql server 获取最近七个月的月份信息

sql 复制代码
WITH recent_dates_cte AS (
    -- 初始查询,获取当前日期的订单记录
    SELECT CAST(GETDATE() AS DATE) AS dayKey

    UNION ALL

    -- 递归查询,获取前一天的日期
    SELECT DATEADD(MONTH, -1, dayKey)
    FROM recent_dates_cte
    WHERE dayKey >= DATEADD(MONTH, -5, CAST(GETDATE() AS DATE))
),
month_key_cte as (
SELECT CONVERT(varchar(7),dayKey,120)  as monKey
FROM recent_dates_cte
)

select * from month_key_cte

sql server 获取最近七周的周信息

sql 复制代码
WITH recent_dates_cte AS (
    -- 初始查询,获取当前日期的订单记录
    SELECT CAST(GETDATE() AS DATE) AS dayKey

    UNION ALL

    -- 递归查询,获取前一天的日期
    SELECT DATEADD(WEEK, -1, dayKey)
    FROM recent_dates_cte
    WHERE dayKey >= DATEADD(WEEK, -5, CAST(GETDATE() AS DATE))
),
week_key_cte as (
    SELECT 
    CONCAT(DATEPART(YEAR, dayKey), '-', FORMAT(DATEPART(WEEK, dayKey), '00')) AS weekKey,
    CASE 
        WHEN DATEDIFF(WEEK, dayKey, GETDATE()) = 0 THEN '本周'
        WHEN DATEDIFF(WEEK, dayKey, GETDATE()) = 1 THEN '前1周'
        WHEN DATEDIFF(WEEK, dayKey, GETDATE()) = 2 THEN '前2周'
        WHEN DATEDIFF(WEEK, dayKey, GETDATE()) = 3 THEN '前3周'
        WHEN DATEDIFF(WEEK, dayKey, GETDATE()) = 4 THEN '前4周'
        WHEN DATEDIFF(WEEK, dayKey, GETDATE()) = 5 THEN '前5周'
        WHEN DATEDIFF(WEEK, dayKey, GETDATE()) = 6 THEN '前6周'
    END AS label
	FROM recent_dates_cte
)

select * from week_key_cte

有了这样的基础,实现最近6个月,最近5年,最近6周是不是特别简单了,方便统计,比如统计最近六周,半年,最近六年,有了with以后是不是好理解多了。

公用表表达式(CTE)是处理复杂查询的强大工具,提供了更好的代码可读性和结构化。在 MySQL 和 SQL Server 中使用 CTE,能够简化复杂的子查询,并且方便处理递归查询和分页查询。通过本文的介绍,希望读者能对 CTE 的概念和用法有更清晰的理解,并能在实际工作中应用这些技巧。

相关推荐
刘大浪几秒前
后端数据增删改查基于Springboot+mybatis mysql 时间根据当时时间自动填充,数据库连接查询不一致,mysql数据库连接不好用
数据库·spring boot·mybatis
无敌岩雀8 分钟前
MySQL中的索引
数据库·mysql
a_安徒生30 分钟前
linux安装TDengine
linux·数据库·tdengine
东阳马生架构32 分钟前
MySQL原理简介—1.SQL的执行流程
mysql
程序员学习随笔33 分钟前
PostgreSQL技术内幕19:逻辑备份工具pg_dump、pg_dumpall
数据库·postgresql
尘浮生1 小时前
Java项目实战II基于微信小程序的校运会管理系统(开发文档+数据库+源码)
java·开发语言·数据库·微信小程序·小程序·maven·intellij-idea
偶尔。5351 小时前
什么是事务?事务有哪些特性?
数据库·oracle
安迁岚1 小时前
【SQL Server】华中农业大学空间数据库实验报告 实验六 视图
数据库·sql·mysql·oracle·实验报告
喵叔哟1 小时前
16. 【.NET 8 实战--孢子记账--从单体到微服务】--汇率获取定时器
微服务·oracle·.net
xoxo-Rachel1 小时前
(超级详细!!!)解决“com.mysql.jdbc.Driver is deprecated”警告:详解与优化
java·数据库·mysql