八个不可不知的SQL高级方法

结构化查询语言(SQL)是一种广泛使用的工具,用于管理和操作数据库。基本的SQL查询简单易学,但掌握高级SQL技术可以将您的数据分析和管理能力提升到新的高度。

高级SQL技术是指一系列功能和函数,使您能够对数据执行复杂操作,例如聚合、连接、子查询、窗口函数和递归查询。

通过深入了解SQL的高级特性和技巧,您可以更有效地进行数据分析和管理,为您的工作带来更大的价值。

本文将详细介绍以下技术,并使用具体且易于理解的示例。

1. 窗口函数

窗口函数支持在与当前行相关的一组行上执行计算,可以根据指定的窗口定义进行聚合、排序和分析操作。这种计算方式可以提供更灵活和精确的数据分析能力。

例如;有一个名为orders的表,其中包含以下列:order_id、customer_id、order_date和order_amount。您想要计算每个客户的销售总额,按其订单日期排序。您可以使用SUM窗口函数来实现这一点:

SELECT order_id, customer_id, order_date, order_amount,SUM(order_amount) OVER (PARTITION BY customer_idORDER BY order_dateROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) AS running_total
FROM orders;

在此示例中,SUM函数应用于order_amount列,并按customer_id列进行分区。这意味着每个客户的累计销售额将分别计算。

ORDER BY子句指定应使用订单日期来确定每个分区内行的顺序。这意味着将按客户订单的顺序计算累计销售额。

ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW子句指定计算的窗口框架应包括从分区开始到当前行为止的所有行。这意味着将从每个客户的第一笔订单开始计算累计销售额,直到包括当前订单。

查询的结果将是一个包含与orders表相同列的表,以及一个名为running_total的附加列,其中包含每个客户的累计销售额,按其订单日期排序。

通过在SQL中使用窗口函数,您可以对数据执行复杂计算,并深入了解业务。此示例演示了如何为每个客户计算累计销售额,但是您可以使用窗口函数执行许多其他类型的计算,例如计算移动平均值、排名数据等。

2. 公共表达式(CTEs)

公共表达式(CTEs)支持您在SQL查询中定义一个临时结果集,并将其命名为一个表,以便在后续的SQL语句中引用和使用该临时结果集。这种方式可以提高查询的可读性和可维护性,并使查询逻辑更加模块化和灵活。

假设有一个名为employees的表,其中包含以下列:employee_id、employee_name、department_id和salary。您想要计算每个部门的平均工资,然后找到其工资高于部门平均工资的员工。为此,您可以使用CTE首先计算每个部门的平均工资,然后在第二个查询中使用该CTE查找其工资高于部门平均工资的员工。

WITH department_avg_salary AS (SELECT department_id, AVG(salary) AS avg_salaryFROM employeesGROUP BY department_id
)SELECT employee_id, employee_name, salary, department_avg_salary.avg_salary
FROM employees
INNER JOIN department_avg_salary ON employees.department_id = department_avg_salary.department_id
WHERE salary > department_avg_salary.avg_salary;

在此示例中,第一个查询定义了一个名为department_avg_salary的CTE。该CTE使用AVG函数和GROUP BY子句计算每个部门的平均工资,将员工按其部门分组。

然后,第二个查询将department_avg_salary CTE视为表格,并将其与employees表格在department_id列上连接。结果由WHERE子句过滤,仅包括其工资高于其部门平均工资的员工。在这种情况下使用CTE的优点在于它允许您将问题分解为两个步骤:首先计算每个部门的平均工资,然后选择其工资高于其部门平均工资的员工。通过将计算分成两个步骤,查询更易于阅读和维护。

CTEs也可以用于许多其他情况,例如递归查询、复杂连接等。通过使用CTE,您可以使SQL查询更易于阅读和理解。

3. 聚合函数

聚合函数是用于对一组值进行计算并返回单个结果值的函数。它们可以在表的多行或多列之间执行计算,并且能够以有意义的方式对数据进行汇总。在SQL中,最常见的聚合函数包括SUM(求和)、AVG(平均值)、MIN(最小值)、MAX(最大值)和COUNT(计数)。这些函数在数据分析和报告中非常实用,可以帮助我们快速获得对数据集的总结统计信息。

例如,有一个名为sales的表,其中包含以下列:sale_id、product_id、sale_date、sale_amount和region。您想要计算每个产品的总销售额和平均销售额,以及每个地区的畅销产品。为此,您可以使用聚合函数按产品和地区分组销售,并计算总销售额和平均销售额,以及找到每个地区的畅销产品。

SELECT product_id, AVG(sale_amount) AS avg_sale_amount, SUM(sale_amount) AS total_sale_amount, region, RANK() OVER (PARTITION BY region ORDER BY SUM(sale_amount) DESC) AS rank
FROM sales
GROUP BY product_id, region;

在此示例中,查询有三个聚合函数:AVG、SUM和RANK。

AVG函数计算每个产品和地区的平均销售额,而SUM函数计算每个产品和地区的总销售额。GROUP BY子句按产品和地区分组销售。

RANK函数用于查找每个地区的畅销产品。OVER子句指定应单独对每个地区进行排名,而PARTITION BY子句指定要对数据进行分区的列(在本例中为region)。ORDER BY子句指定排名应基于每个地区中每种产品的销售金额总和,并按降序排列。

查询结果包括product_id、region、total_sale_amount、avg_sale_amount和rank列。rank列指示每个地区中每种产品基于总销售额的排名,排名第一的畅销产品在每个地区都是1。

使用聚合函数,在这种情况下,优点在于它们允许您对数据进行分组和汇总,并计算有用的指标,例如总销售额和平均销售额。RANK函数还允许您查找每个地区的畅销产品,这对于识别趋势和改进机会非常有用。

聚合函数也可以用于许多其他情况,例如计算最小值和最大值、计数记录等。通过使用聚合函数,您可以使SQL查询更加强大和灵活。

4. 透视表

透视表是一种用于从较大的表格中进行数据汇总和聚合,以便更方便进行分析的表格。它可以将数据从行转换为列,并以更有意义的方式展示数据。

在SQL中,我们可以使用PIVOT运算符创建透视表。该运算符可以根据指定的列对数据进行汇总,并以表格的形式呈现结果,使数据更易于理解和分析。透视表为我们提供了一种灵活且直观的方式来汇总和展示数据,从而帮助我们更好地理解数据的关系和趋势。

例如:

SELECT customer_id, [1] AS Product1, [2] AS Product2, [3] AS Product3, [4] AS Product4, [5] AS Product5
FROM (SELECT customer_id, product_id, order_quantityFROM orders
) p
PIVOT (SUM(order_quantity)FOR product_id IN ([1], [2], [3], [4], [5])
) AS pvt;

在上面的示例中,查询使用PIVOT运算符按产品ID旋转数据,每个客户都有一个列。SUM函数用于计算每个客户订购的每种产品的总数量。

子查询p用于从orders表中提取必要的列。然后将PIVOT运算符应用于子查询,使用SUM函数计算每个客户订购的每种产品的总数量。FOR子句指定了透视列(在本例中为product_id),而IN子句指定了要透视的值(在本例中为[1]、[2]、[3]、[4]、[5])。

查询的结果是一个透视表,显示了每个客户订购的每种产品的总数量,每种产品都有一列,每个客户都有一行。

透视表也可以用于许多其他情况,例如汇总销售数据、分析调查结果等。通过使用透视表,您可以使SQL查询更加强大和灵活。

5. 子查询

SQL中的子查询是用于从一个或多个表中检索数据的嵌套查询,子查询的结果在主查询中使用。它们可用于过滤、排序和分组数据,并可分类为单行或多行子查询。子查询括在括号中,可以在SQL语句的各个部分中使用,例如SELECT、FROM、WHERE和HAVING子句。

例如;有两个名为employees和salaries的表。employees表包含以下列:employee_id、first_name、last_name和department_id。salaries表包含以下列:employee_id、salary和salary_date。您想要找到每个部门工资最高的员工的姓名。为此,您可以使用子查询查找每个部门的最高工资,然后将结果与employees和salaries表连接以获取具有该工资的员工的姓名。

下面的查询使用子查询查找每个部门的最高工资。首先执行子查询并返回包含每个部门最高工资的结果集。然后,主查询将employees和salaries表与子查询的结果连接,以获取每个部门工资最高的员工的姓名。

SELECT e.first_name, e.last_name, e.department_id, s.salary
FROM employees e INNER JOIN salaries s ON e.employee_id = s.employee_id INNER JOIN (SELECT department_id, MAX(salary) AS max_salaryFROM salariesGROUP BY department_id) m ON s.department_id = m.department_id AND s.salary = m.max_salary;

使用INNER JOIN子句将employees和salaries表连接起来,使用employee_id列作为连接键。使用department_id列将子查询连接到主查询,并使用salary列匹配每个部门的最高工资。

查询的结果是一个表格,显示每个部门工资最高的员工的姓名及其部门ID和工资。

6. 交叉连接

交叉连接是一种连接操作,用于返回两个或多个表的所有可能行组合,而不需要连接条件。它在生成测试数据或需要获取多个表格所有可能组合的计算时非常有用。然而,由于交叉连接可能会产生高计算成本和庞大的结果集,因此在使用时需要谨慎考虑其影响,并确保结果集的大小符合预期。通常情况下,应优先考虑使用其他类型的连接操作,如内连接、外连接或等值连接,以更有效地获取所需的数据。

在下面的示例中,有两个名为customers和orders的表。customers表包含以下列:customer_id、customer_name和city。orders表包含以下列:order_id、customer_id和order_date。您想要找到每个客户在每个城市下单的总数。为此,您可以使用交叉连接生成一个结果集,将每个客户与每个城市组合,然后将结果与orders表连接以获取每种组合的订单数量。

SELECT c.customer_id, c.customer_name, c.city, COUNT(o.order_id) AS order_count
FROM customers c CROSS JOIN (SELECT DISTINCT cityFROM customers) cities LEFT JOIN orders o ON c.customer_id = o.customer_id
WHERE c.city = cities.city
GROUP BY c.customer_id, c.customer_name, c.city;

示例中,查询使用交叉连接生成一个结果集,该结果集将每个客户与每个城市组合在一起。交叉连接首先执行,返回一个包含每个客户和城市的每个组合的结果集。然后,主查询使用左连接将交叉连接的结果与orders表连接,以确保即使客户没有下订单也包括所有客户在结果中。

WHERE子句用于过滤结果,仅包括客户所在城市与交叉连接中的城市匹配的行。这确保了结果仅显示每个客户在其各自城市中的订单数量。

GROUP BY子句用于按客户ID、客户名称和城市分组结果。COUNT()函数用于计算每个客户在每个城市中的订单数量。

查询的结果是一个表格,显示了每个客户在每个城市中下达的订单总数。

7. 临时表

SQL中的临时表是在执行SQL语句或事务期间创建和使用的表。它们存储在内存或磁盘上,并在创建它们的会话结束或不再需要时自动删除。临时表通常用于存储中间结果,或将复杂查询分解为更小、更易管理的部分。

它们可以使用CREATE TEMPORARY TABLE语句创建,并像常规表一样使用SQL命令(如SELECT、INSERT、UPDATE和DELETE)进行操作。临时表可以非常有用,用于优化复杂查询并提高性能,因为它们可以帮助减少需要在任何给定时间处理的数据量。

假设有一个名为sales的表,其中包含以下列:date、product、category和sales_amount。您想创建一个报告,显示过去一年每个月每个类别的总销售额。为此,您可以使用一个临时表来创建每个月销售数据的摘要,然后将临时表与sales表连接,以获取每个类别的总销售额。

首先,您可以使用CREATE TEMPORARY TABLE语句创建临时表:

CREATE TEMPORARY TABLE monthly_sales_summary (month DATE,category VARCHAR(50),total_sales DECIMAL(10,2)
);

此语句创建了一个名为monthly_sales_summary的临时表,其中包含三列:month、category和total_sales。month列的类型为DATE,category列的类型为VARCHAR(50),total_sales列的类型为DECIMAL(10,2)。

接下来,使用INSERT INTO语句将摘要数据填充到临时表中:

INSERT INTO monthly_sales_summary (month, category, total_sales)
SELECT DATE_TRUNC('month', date) AS month,category,SUM(sales_amount) AS total_sales
FROM sales
WHERE date >= DATE_TRUNC('year', CURRENT_DATE) -- sales from the past year
GROUP BY DATE_TRUNC('month', date),category;

此语句使用DATE_TRUNC函数将date列截断到月份级别,按月份和类别分组销售数据。此查询的结果插入到monthly_sales_summary表中,该表现在包含每个月销售数据的摘要。

最后,可以将临时表与sales表连接起来,以获取每个类别的总销售额:

SELECT s.category, mss.month, mss.total_sales
FROM sales s JOIN monthly_sales_summary mss ON s.category = mss.category AND DATE_TRUNC('month', s.date) = mss.month
WHERE s.date >= DATE_TRUNC('year', CURRENT_DATE) -- sales from the past year
ORDER BY s.category, mss.month;

此语句将sales表与monthly_sales_summary表连接在category和month列上,并从临时表中选择category、month和total_sales列。WHERE子句用于过滤结果,仅包括过去一年的销售数据,ORDER BY子句用于按类别和月份对结果进行排序。

查询的结果是一个表格,显示了过去一年每个月每个类别的总销售额。

8. 具体化视图

SQL中的具体化视图是存储为物理表的预计算结果集。它们基于SQL查询创建和维护,并用于提高频繁执行查询的性能。具体化视图可以按计划或按需刷新,以确保数据是最新的。当针对具体化视图执行查询时,结果集从物理表中检索,而不是从原始表中计算。

这可以提升性能,特别是对于涉及联接或聚合函数的复杂查询。具体化视图通常用于数据仓库和业务智能应用程序中,在这些应用程序中,它们可以帮助加速报告和仪表板。

例如,有一个名为sales的大型表,其中包含以下列:date、product、category和sales_amount。您想创建一个报告,显示过去一年每个月每个类别的总销售额。但是,直接在sales表上运行此查询会很慢,因为它包含数百万行。为了加快查询速度,可以创建一个物化视图,按月份和类别汇总销售数据。

要创建物化视图,可以使用CREATE MATERIALIZED VIEW语句,如下所示:

CREATE MATERIALIZED VIEW monthly_sales_summary AS 
SELECT DATE_TRUNC('month', date) AS month,category,SUM(sales_amount) AS total_sales
FROM sales
WHERE date >= DATE_TRUNC('year', CURRENT_DATE) -- sales from the past year
GROUP BY DATE_TRUNC('month', date),category;

此语句创建了一个名为monthly_sales_summary的物化视图,其中包含每个月和类别的销售数据摘要。SELECT语句与前面示例中用于创建临时表的语句相同,但是不使用临时表,而是将结果存储在物化视图中。

物化视图与表类似,因为它们将数据存储在磁盘上,但是在基础数据更改时会自动更新。您可以使用REFRESH MATERIALIZED VIEW语句手动刷新物化视图,也可以使用cron作业或其他调度工具设置定期刷新。

创建物化视图后,可以像查询其他表一样查询它:

SELECT category, month, total_sales
FROM monthly_sales_summary
ORDER BY category, month;

此语句从monthly_sales_summary物化视图中选择category、month和total_sales列,并按类别和月份对结果进行排序。

在这种情况下使用物化视图的优点在于,它允许您预计算和存储摘要数据,从而减少运行查询所需的时间。物化视图特别适用于经常运行并需要对大型数据集进行复杂计算的报告。但是,它们也有一些限制,例如它们可能占用大量磁盘空间,并且在基础数据更改时可能无法立即更新。

结语

掌握高级SQL技术,如窗口函数、CTE、聚合函数、透视表、子查询、交叉连接、临时表和物化视图,可以帮助您更有效地处理复杂的数据分析任务。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/146324.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

记录:Unity脚本的编写

目录 前言添加脚本到unity编写c#脚本查看效果 前言 在学习软件构造这门课的时候,对unity和c#进行了 一定程度的学习,包括简单的建立地形,添加对象,添加材质等,前不久刚好学习了如何通过c#脚本对模型进行操控&#xff…

uniapp - 微信小程序实现腾讯地图位置标点展示,将指定地点进行标记选点并以一个图片图标展示出来(详细示例源码,一键复制开箱即用)

效果图 在uniapp微信小程序平台端开发,简单快速的实现在地图上进行位置标点功能,使用腾讯地图并进行标点创建和设置(可以自定义标记点的图片)。 你只需要复制代码,改个标记图标和位置即可。

Fiddler Orchestra用户指南:打造高效协同调试利器

引言:今天Fiddler更新到5.0版本后,小酋不经意间晃到了“Fiddler Orchestra”选项卡。爱折腾的小酋赶紧链接到官方用户指南一睹为快,看看这是什么东西,实现了什么新功能。下面是小酋看后做的一个翻译抢先版。 这是了解和设置Fiddl…

《 新手》web前端(axios)后端(java-springboot)对接简解

文章目录 <font color red>1.何为前后端对接?2.对接中关于http的关键点2.1. 请求方法2.2. 请求参数设置简解&#xff1a; 3.对接中的跨域(CROS)问题**为什么后端处理跨域尽量在业务之前进行&#xff1f;**3.总结 1.何为前后端对接? “前后端对接” 是指前端和后端两个…

ElementUI实现增删改功能以及表单验证

目录 前言 BookList.vue action.js 展示效果 前言 本篇还是在之前的基础上&#xff0c;继续完善功能。上一篇完成了数据表格的查询&#xff0c;这一篇完善增删改&#xff0c;以及表单验证。 BookList.vue <template><div class"books" style"pa…

picoctf_2018_can_you_gets_me

picoctf_2018_can_you_gets_me Arch: i386-32-little RELRO: Partial RELRO Stack: No canary found NX: NX enabled PIE: No PIE (0x8048000)32位&#xff0c;只开了NX 拿到这么大的程序&#xff0c;直接ROPchain看看 #!/usr/bin/env python2# execve …

低代码工作流程管理系统:提升企业运营效率的利器

业务运营状况是否良好&#xff0c;除了人员需要配合以外&#xff0c;真正发挥作用的是背后的工作流程。将重复的工作进行自动化处理&#xff0c;确保这些流程最终指向同一个目标、实现一致的运营结果。而设计和实施不佳的工作流程则产生相反的效果——导致处理时间延长、运营成…

好题分享

1.Problem - G - Codeforces &#xff08;1&#xff09;题意 &#xff08;2&#xff09;思路 因为最多13次&#xff0c;那么不如我们就问13次&#xff0c;然后考虑把每一个位置重新按二进制拆分成一个下标&#xff0c;因为C(13,6) > 1000,因此在数量上是满足得&#xff0c;我…

编程每日一练(多语言实现)基础篇:满足abcd=(ab+cd)^2的数 (增加Go语言实现)

文章目录 一、实例描述二、技术要点三、代码实现3.1 C 语言实现3.2 Python 语言实现3.3 Java 语言实现3.4 JavaScript 语言实现3.5 Go 语言实现 一、实例描述 假设 abcd 是一个四位整数&#xff0c;将它分成两段&#xff0c;即 ab 和 cd&#xff0c;使之相加求和后再平方。求满…

LeetCode 热题 HOT 100:回溯专题

LeetCode 热题 HOT 100&#xff1a;https://leetcode.cn/problem-list/2cktkvj/ 文章目录 17. 电话号码的字母组合22. 括号生成39. 组合总和46. 全排列补充&#xff1a;47. 全排列 II &#xff08;待优化)78. 子集79. 单词搜索124. 二叉树中的最大路径和200. 岛屿数量437. 路径…

【C++】C++的类型转换

文章目录 1. C语言中的类型转换2. C中的类型转换2.1 static_cast2.2 reinterpret_cast2.3 const_cast2.4 dynamic 1. C语言中的类型转换 在C语言中&#xff0c;经常会出现一种情况&#xff1a;运算符两边的类型不同&#xff0c;或者形参实参类型不匹配&#xff0c;此时就会发生…

工信部:杭州亚运会开幕式首创 5G 超密组网方案,场馆网络无缝覆盖

“工信 V 报”今日发布消息称&#xff0c;工信部经过精心统筹、周密部署&#xff0c;举全系统之力圆满完成了杭州亚运会开幕式各项保障任务。 据介绍&#xff0c;亚运会的指挥调度、安全保卫、通信网络、计时记分、电视转播等系统顺畅运行&#xff0c;对无线电安全、信息通信服…

MATLAB与Python:优势与挑战

本文旨在探讨MATLAB与Python在特定领域内的使用情况&#xff0c;并分析两者之间的优势和挑战。 MATLAB和Python都是流行的编程语言&#xff0c;广泛应用于科学计算、数据分析和机器学习等领域。在某些领域&#xff0c;如航空航天工程、自动化和电子工程嵌入式系统开发等&#…

福建江夏学院蔡慧梅主任一行莅临拓世科技集团,共探AI+时代教育新未来

在科技的海洋中&#xff0c;产业是那航行的巨轮&#xff0c;而教育则是指引方向的灯塔。当巨轮与灯塔相互辉映&#xff0c;产教融合与校企合作便成为了推动国家科技创新和人才培养的金钥匙&#xff0c;为未来开启一扇扇充满希望的大门。 2023年9月24日&#xff0c;福建江夏学院…

Nginx简介与Docker Compose部署指南

Nginx是一款高性能的开源Web服务器和反向代理服务器&#xff0c;以其卓越的性能、可伸缩性和灵活性而闻名。它在全球范围内广泛用于托管Web应用程序、负载均衡、反向代理和更多场景中。在本文中&#xff0c;我们将首先介绍Nginx的基本概念&#xff0c;然后演示如何使用Docker C…

Go结构体深度探索:从基础到应用

在Go语言中&#xff0c;结构体是核心的数据组织工具&#xff0c;提供了灵活的手段来处理复杂数据。本文深入探讨了结构体的定义、类型、字面量表示和使用方法&#xff0c;旨在为读者呈现Go结构体的全面视角。通过结构体&#xff0c;开发者可以实现更加模块化、高效的代码设计。…

doT.js模板学习笔记

doT.js模板学习笔记 欢迎学习doT.js模板学习笔记doT.js模板是什么doT.js 主要优势在doT.js好处引入方式基本语法语法示例结尾 欢迎学习doT.js模板学习笔记 doT.js官方网站 本文章得示例源码 doT.js模板是什么 doT.js 是一个 JavaScript 模板框架&#xff0c;在 web 前端使用 d…

软件测试面试复盘

作者&#xff1a;爱塔居 专栏&#xff1a;测试 1、计算机网络七层协议&#xff1a;物理层、数据链路层、网络层、传输层、表示层、会话层、应用层&#xff08;面试问过这个&#xff09; 2.TCP/IP四层模型&#xff1a;应用层、传输层、网络层、网络接口层&#xff08;笔试问过&…

vscode左键无法跳转到定义的文件

之前用vscode的时候&#xff0c;明明是可以ctrl键鼠标左键跳转到定义文件的&#xff0c;突然之间就不行了&#xff0c;鼠标移到引入上根本都没有下划线&#xff0c;无法跳转 解决方法&#xff1a; 项目的根目录新建 jsconfig.json 文件&#xff0c;代码如下 {"compiler…

使用sqlmap总是提示需要302跳转重新登录的解决方法

如果在命令中不指定cookie&#xff0c;sqlmap在执行时会提示需要重新登录 如果给了cookie但发现还是提示需要重新登录&#xff0c;且按它给的提示发现还是找不到注入点&#xff0c;原因是url没有加引号 url加了双引号后解决问题