我的编程空间,编程开发者的网络收藏夹
学习永远不晚

SQL 查询之公用表表达式(CTE)

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

SQL 查询之公用表表达式(CTE)

CTE 可以简化查询语句

CTE的基础语法如下。

1WITH my_cte AS (
2  SELECT a,b,c
3  FROM T1
4)
5SELECT a,c
6FROM my_cte
7WHERE ....

CTE 语句以 WITH 关键字开始,所以 CTE 又被称为 WITH 语句。在 WITH 关键字之后是 CTE 的名字;在 AS 关键字之后的括号里写入要实现的查询语句。

在上例中,CTE 的名字是 my_cte,查询语句为 SELECT a,b,c FROM T1。

主查询在定义 CTE 语句的括号后面。定义 CTE 之后,即可在主查询中通过 CTE 的名字对其进行引用。主查询也称为外部查询,在这个示例中是 SELECT a,c FROM my_cte WHERE ...。

简单的 CTE 示例

本文中的例子都基于以下这个 sales 表,这个表模拟了一家连锁书店的销售数据,示例如下。

示例数据

生成 sales 表的 SQL 代码如下。

1-- 创建名为sales的表
 2CREATE TABLE sales (
 3    branch VARCHAR(50),
 4    date DATE,
 5    seller VARCHAR(50),
 6    item VARCHAR(50),
 7    quantity INT,
 8    unit_price DECIMAL(10, 2)
 9);
10
11-- 插入数据
12INSERT INTO sales (branch, date, seller, item, quantity, unit_price)
13VALUES
14    ('Beijing-1', '2024-06-07', 'Zhangsan', 'SQL Data Analysis', 1, 80),
15    ('Shanghai-1', '2024-06-06', 'Lisi', 'Python Data Analysis', 2, 120),
16    ('Shanghai-2', '2024-06-07', 'Wangwu', 'Pandas Workshop', 1, 60),
17    ('Beijing-1', '2024-06-07', 'Zhangsan', 'Polars in Action', 1, 50),
18    ('Shanghai-2', '2024-06-07', 'Wangwu', 'Echarts Data Charts', 2, 90),
19    ('Shanghai-1', '2024-06-07', 'Lisi', 'Mastering Pandas', 5, 75),
20    ('Shanghai-1', '2024-06-07', 'Zhaoliu', 'R Projects', 2, 100);

我们先看一个简单的例子。输出 sales 表中的原始数据记录,但要在 sales 表中再添加一列,显示当天(date)在同一分店中售出图书的最高价格。获取售价最高的图书价格可以使用如下 CTE。

1WITH highest AS (
 2  SELECT
 3    branch,
 4    date,
 5    MAX(unit_price) AS highest_price
 6  FROM sales
 7  GROUP BY branch, date
 8)
 9SELECT
10  sales.*,
11  h.highest_price
12FROM sales
13JOIN highest h
14  ON sales.branch = h.branch
15    AND sales.date = h.date

上述查询定义了名为 highest 的 CTE,这个虚拟表包含的列为 branch、date 和highest_price,分别表示分店的名称、销售日期和当天该分店销售的最高单价。

然后,在主查询中,像使用常规表一样使用虚拟表 highest。

最后,将 highest 的查询结果与 sales 表进行连接。

查询的结果如下。

查询结果

下面这个例子显示各分店最高的单日收入。

1WITH daily_revenue AS (
 2  SELECT
 3    branch,
 4    date,
 5    SUM(unit_price * quantity) AS daily_revenue
 6  FROM   sales
 7  WHERE EXTRACT(YEAR FROM date) = 2024
 8  GROUP BY 1,2
 9)
10SELECT
11  branch,
12  MAX(daily_revenue) max_daily_revenue
13FROM daily_revenue
14GROUP BY 1
15ORDER BY 2 DESC

在这段代码中,首先定义名为 daily_revenue 的 CTE,包含的列为 branch、date 和 daily_revenue,数据记录为每家分店在 2024 年内每一天的日销售额。

然后,在主查询中,获取 2024 年每家分店的最高日销售额。

最后,按max_daily_revenue 降序排列输出数据。

查询结果

在 SQL 高级查询中使用 CTE

主查询中可以使用多个 CTE。下面的示例中介绍如何使用多个 CTE 对复杂的 SQL 语句进行简化。

假设要显示某个城市的月销售额,以及该城市中每个分店的销售额。

在本例中,先创建两个 CTE。再在主查询中把这两个 CTE 连接起来。

1WITH shanghai1_monthly_revenue AS (
 2  SELECT
 3    EXTRACT(MONTH FROM date) as month,
 4    SUM(unit_price * quantity) AS revenue
 5  FROM sales
 6  WHERE EXTRACT(YEAR FROM date) = 2024
 7    AND branch = 'Shanghai-1'
 8  GROUP BY 1
 9),
10shanghai2_monthly_revenue AS (
11  SELECT
12    EXTRACT(MONTH FROM date) as month,
13    SUM(unit_price * quantity) AS revenue
14  FROM sales
15  WHERE EXTRACT(YEAR FROM date) = 2024
16    AND branch = 'Shanghai-2'
17  GROUP BY 1
18)
19SELECT
20  s1.month,
21  s1.revenue + s2.revenue AS shanghai_revenue,
22  s1.revenue AS shanghai1_revenue,
23  s2.revenue AS shanghai2_revenue
24FROM shanghai1_monthly_revenue s1, shanghai2_monthly_revenue s2
25WHERE s1.month = s2.month

在上述代码中,定义了 shanghai1_monthly_revenue 与 shanghai2_monthly_revenue 两个 CTE,使用这两个 CTE 获取 2024 年上海各分店的月销售额。

接下来,使用 month 列连接这两个 CTE,并把两个分店的销售额相加,以获取上海的总销售额。

查询结果如下。

查询结果

下例显示每家分店最大金额的订单及该订单的日期。为此,需要创建一个 CTE,根据订单金额为各分店的订单进行排名(position 列就是排名)。

1WITH tickets AS (
 2  SELECT distinct
 3    branch,
 4    date,
 5    unit_price * quantity AS ticket_amount,
 6    ROW_NUMBER() OVER (
 7      PARTITION BY branch
 8      ORDER by unit_price * quantity DESC
 9    ) AS position
10  FROM sales
11  ORDER BY 3 DESC
12)
13SELECT
14  branch,
15  date,
16  ticket_amount
17FROM tickets
18WHERE position =1

在这段代码中,创建了名为 tickets 的 CTE,包含 branch、date、titcket_amount和 position。

在主查询中,过滤 position 等于 1 的记录,以获取各分店销售额最大的订单。

查询结果如下。

查询结果

SQL 查询中的嵌套 CTE

本例介绍如何使用嵌套 CTE。输出所有单价超过 90 元的图书,以及上海分店(Shanghai-2)的销售数量。

1WITH over_90_items AS (
 2  SELECT DISTINCT
 3    item,
 4    unit_price
 5  FROM sales
 6  WHERE unit_price >=90
 7),
 8shanghai2_over_90 AS (
 9  SELECT
10    o90.item,
11    o90.unit_price,
12    coalesce(SUM(s.quantity), 0) as total_sold
13  FROM over_90_items o90
14  LEFT JOIN sales s
15  ON o90.item = s.item AND s.branch = 'Shanghai-2'
16  GROUP BY o90.item, o90.unit_price
17)
18SELECT item, unit_price, total_sold
19FROM shanghai2_over_90;

over_90_items 用于筛选单价大于或等于 90 元的图书。

shanghai2_over_90 用于筛选 Shanghai-2 分店售出的超过 90 元的图书的数量。

这就是嵌套 CTE 的基本用法。

注意,shanghai2_over_90 中的 FROM 子句中引用了 over_90_items。

使用 LEFT JOIN sales 的原因是 Shanghai-2 分店可能没有销售过价格超过 90 元的图书。

查询结果如下。

查询结果

递归查询与 CTE

在关系型数据库中,表示数据层级结构的表很常见,例如,上下级关系、父子组件关系等。

为了在这些层级结构中进行从上到下或从下到上的遍历,可以使用递归 CTE。

RECURSIVE 是定义遍历递归数据结构的 CTE 的关键字。CTE 递归查询的形式如下。

1WITH RECURSIVE  cte_name AS (
2     CTE_query_definition  -- 非递归查询项
3UNION ALL
4     CTE_query_definition  -- 递归查询项
5)
6SELECT * FROM cte_name;

限于本文的篇幅,有 CTE 的递归查询就先不赘述了,关注我们的公众号,我会在后续的文章中进行介绍。

结语

SQL 的 CTE 非常强大,它可以让 SQL 查询代码更简洁,逻辑更清晰。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

SQL 查询之公用表表达式(CTE)

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

SQL 查询之公用表表达式(CTE)

在关系型数据库中,表示数据层级结构的表很常见,例如,上下级关系、父子组件关系等。为了在这些层级结构中进行从上到下或从下到上的遍历,可以使用递归 CTE。

MySQL 递归 CTE(公用表表达式)

MySQL 中的递归 CTE(公用表表达式)是一种处理递归查询的方法。CTE 允许您在查询中创建临时表,并在查询中引用该临时表。以递归查询为例,假设有一个员工表,其中包含员工的 ID、姓名和经理的 ID。现在需要找出某个员工的所有下属(直接
2023-10-20

SqlServer公用表表达式(CTE)的具体使用

目录1、本文内容2、语法3、参数4、创建和使用公用表表达式的准则5、定义和使用递归公用表表达式的准则6、示例6.1、下例显示每名销售代表每年的销售订单总数。6.2、使用公用表表达式来限制次数和报告平均数6.3、在单个查询中使用多个 CTE
SqlServer公用表表达式(CTE)的具体使用
2024-08-16

MariaDB表中的公用表表达式CTE怎么理解

今天就跟大家聊聊有关MariaDB表中的公用表表达式CTE怎么理解,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。前言公用表表达式(Common Table Expression,CT
2023-06-29

SQL Server怎么使用T-SQL公用表表达式

本文小编为大家详细介绍“SQL Server怎么使用T-SQL公用表表达式”,内容详细,步骤清晰,细节处理妥当,希望这篇“SQL Server怎么使用T-SQL公用表表达式”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知
2023-06-30

sql server数据库性能优化之2-避免使用CTE公用表达式的递归【by zhang502219048】

数据库优化中的一个实例,记录一下: 1. 原来用了CTE公用表达式的递归,reads高达约40万,看查询执行计划,使用了Nested Loops; 2. 优化去掉递归,改用其它方式实现,reads降低到2639,看查询执行计划,避免了使用Nested Loop
sql server数据库性能优化之2-避免使用CTE公用表达式的递归【by zhang502219048】
2017-05-24

LINQ查询表达式怎么用

这篇文章主要介绍了LINQ查询表达式怎么用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。from子句负责指定LINQ查询操作中的数据源和范围变量1.每一个LINQ查询表达式都
2023-06-17

MySQL 正则表达式查询

MySQL REGEXP:正则表达式正则表达式主要用来查询和替换符合某个模式(规则)的文本内容。例如,从一个文件中提取电话号码,查找
2023-02-03

c#中查询表达式GroupBy怎么用

这篇文章主要介绍c#中查询表达式GroupBy怎么用,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!实验基础数据用例:Student类:public class Student { public i
2023-06-25

C#表达式中的动态查询

当您使用LINQ来处理数据库时,这种体验是一种神奇的体验,对吗?你把数据库实体像一个普通的收集,使用Linq中像Where,Select或者 Take,这些简单的使用就能让代码可用了。
C#动态查询2024-12-02

LINQ查询表达式举例分析

这篇文章主要讲解了“LINQ查询表达式举例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“LINQ查询表达式举例分析”吧!LINQ使查询成为了.NET中头等的编程概念,被查询的数据可以是X
2023-06-17

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录