热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

MySQLWITH语句详解及其应用实例总结

本文将深入探讨MySQL8.0中的WITH语句,通过具体应用实例详细解析其在处理复杂SQL查询时的优势。WITH语句能够显著减少临时表的使用,提高代码的可读性和维护性,从而优化查询性能。

备注:测试数据库版本为MySQL 8.0

这个blog我们来聊聊MySQL的with语句

对于逻辑复杂的sql,with可以大大减少临时表的数量,提升代码的可读性、可维护性

MySQL 8.0终于开始支持with语句了,对于复杂查询,可以不用写那么多的临时表了。

如需要scott用户下建表及录入数据语句,可参考:

scott建表及录入数据sql脚本

语句结构:

with subquery_name1 as (subquery_body1),

subquery_name2 as (subquery_body2)

...

select * from subquery_name1 a, subquery_name2 b

where a.col = b.col

...

优势

– 代码模块化

– 代码可读性增强

– 相同查询唯一化

一.提升代码的可读性和可维护性

需求:求每个部门的平均工资,以及剔除薪资低于1000的实习人员之后的平均工资

-- 求每个部门的平均工资,以及剔除薪资低于1000的实习人员之后的平均工资

-- 主查询的from后面跟了2个临时表,程序可读性不佳

select d.deptno, tmp1.avg_sal avg_sal1, tmp2.avg_sal avg_sal2

from dept d

left join (select e1.deptno, round(avg(ifnull(e1.sal, 0)), 2) avg_sal

from emp e1

group by e1.deptno) tmp1

on d.deptno = tmp1.deptno

left join (select e1.deptno, round(avg(ifnull(e1.sal, 0)), 2) avg_sal

from emp e1

where e1.sal > 1000

group by e1.deptno) tmp2

on d.deptno = tmp2.deptno;

-- 求每个部门的平均工资,以及剔除薪资低于1000的实习人员之后的平均工资

-- 2个临时表的定时语句通过with封装成子查询了,程序可读性增强

with tmp1 as

(select e1.deptno, round(avg(ifnull(e1.sal, 0)), 2) avg_sal

from emp e1

group by e1.deptno),

tmp2 as

(select e1.deptno, round(avg(ifnull(e1.sal, 0)), 2) avg_sal

from emp e1

where e1.sal > 1000

group by e1.deptno)

select d.deptno, tmp1.avg_sal avg_sal1, tmp2.avg_sal avg_sal2

from dept d

left join tmp1

on d.deptno = tmp1.deptno

left join tmp2

on d.deptno = tmp2.deptno;

mysql> -- 求每个部门的平均工资,以及剔除薪资低于1000的实习人员之后的平均工资

mysql> -- 主查询的from后面跟了2个临时表,程序可读性不佳

mysql> select d.deptno, tmp1.avg_sal avg_sal1, tmp2.avg_sal avg_sal2

-> from dept d

-> left join (select e1.deptno, round(avg(ifnull(e1.sal, 0)), 2) avg_sal

-> from emp e1

-> group by e1.deptno) tmp1

-> on d.deptno = tmp1.deptno

-> left join (select e1.deptno, round(avg(ifnull(e1.sal, 0)), 2) avg_sal

-> from emp e1

-> where e1.sal > 1000

-> group by e1.deptno) tmp2

-> on d.deptno = tmp2.deptno;

+--------+----------+----------+

| deptno | avg_sal1 | avg_sal2 |

+--------+----------+----------+

| 10 | 2916.67 | 2916.67 |

| 20 | 2175.00 | 2518.75 |

| 30 | 1566.67 | 1690.00 |

| 40 | NULL | NULL |

+--------+----------+----------+

4 rows in set (0.00 sec)

mysql>

mysql>

mysql> -- 求每个部门的平均工资,以及剔除薪资低于1000的实习人员之后的平均工资

mysql> -- 2个临时表的定时语句通过with封装成子查询了,程序可读性增强

mysql> with tmp1 as

-> (select e1.deptno, round(avg(ifnull(e1.sal, 0)), 2) avg_sal

-> from emp e1

-> group by e1.deptno),

-> tmp2 as

-> (select e1.deptno, round(avg(ifnull(e1.sal, 0)), 2) avg_sal

-> from emp e1

-> where e1.sal > 1000

-> group by e1.deptno)

-> select d.deptno, tmp1.avg_sal avg_sal1, tmp2.avg_sal avg_sal2

-> from dept d

-> left join tmp1

-> on d.deptno = tmp1.deptno

-> left join tmp2

-> on d.deptno = tmp2.deptno;

+--------+----------+----------+

| deptno | avg_sal1 | avg_sal2 |

+--------+----------+----------+

| 10 | 2916.67 | 2916.67 |

| 20 | 2175.00 | 2518.75 |

| 30 | 1566.67 | 1690.00 |

| 40 | NULL | NULL |

+--------+----------+----------+

4 rows in set (0.00 sec)

mysql>

二.with递归

用with递归构造数列

-- 用with递归构造1-10的数据

with RECURSIVE c(n) as

(select 1 union all select n &#43; 1 from c where n <10)

select n from c;

-- 用with递归构造1-10的数据

mysql> with RECURSIVE c(n) as

-> (select 1 union all select n &#43; 1 from c where n <10)

-> select n from c;

&#43;------&#43;

| n |

&#43;------&#43;

| 1 |

| 2 |

| 3 |

| 4 |

| 5 |

| 6 |

| 7 |

| 8 |

| 9 |

| 10 |

&#43;------&#43;

10 rows in set (0.00 sec)

用with递归构造级联关系

with RECURSIVE emp2(ename,empno,mgr,lvl)

as

(select ename, empno, mgr, 1 lvl from emp where mgr is null

union all

select emp.ename, emp.empno, emp.mgr, e2.lvl&#43;1

from emp, emp2 e2

where emp.mgr &#61; e2.empno

)

select lvl,

concat(repeat(&#39;**&#39;,lvl),ename) nm

from emp2

order by lvl,ename

;

mysql> with RECURSIVE emp2(ename,empno,mgr,lvl)

-> as

-> (select ename, empno, mgr, 1 lvl from emp where mgr is null

-> union all

-> select emp.ename, emp.empno, emp.mgr, e2.lvl&#43;1

-> from emp, emp2 e2

-> where emp.mgr &#61; e2.empno

-> )

-> select lvl,

-> concat(repeat(&#39;**&#39;,lvl),ename) nm

-> from emp2

-> order by lvl,ename

-> ;

&#43;------&#43;---------------&#43;

| lvl | nm |

&#43;------&#43;---------------&#43;

| 1 | **KING |

| 2 | ****BLAKE |

| 2 | ****CLARK |

| 2 | ****JONES |

| 3 | ******ALLEN |

| 3 | ******FORD |

| 3 | ******JAMES |

| 3 | ******MARTIN |

| 3 | ******MILLER |

| 3 | ******SCOTT |

| 3 | ******TURNER |

| 3 | ******WARD |

| 4 | ********ADAMS |

| 4 | ********SMITH |

&#43;------&#43;---------------&#43;

14 rows in set (0.00 sec)



推荐阅读
  • 在第六章中,我们将深入探讨MySQL中的多表查询技术,包括联结查询和子查询。联结查询通过将两个或多个表进行连接,基于连接条件生成结果集。常见的联结类型有内联结、外联结和全外联结。交叉联结(CROSS JOIN)虽然使用较少,但其原理是生成所有可能的组合,类似于笛卡尔积的概念。此外,子查询则是在一个查询语句中嵌套另一个查询,用于获取更复杂的数据集。本章将通过实例详细讲解这些查询方法的应用和优化技巧。 ... [详细]
  • 本文提供了在Windows系统上部署和启动MySQL免安装版本的详细步骤。首先,从MySQL官方网站下载社区版免安装包(https://dev.mysql.com/downloads/mysql/8.0.html),将其解压至指定目录,例如D:\tools\mysql。接着,配置系统环境变量,确保MySQL命令行工具可以在任意路径下使用。此外,还需创建并配置my.ini文件以设置MySQL的基本参数,确保数据库服务能够顺利启动和运行。 ... [详细]
  • 本周,我深入研究了 ECharts 插件的使用方法,整体感觉插件操作较为简便,但后台算法较为复杂。此外,我还学习了 MySQL 函数的新应用,进一步提升了数据库操作的灵活性。同时,分享了自己在 Python 书籍外借过程中的体验,总结了一些实用的借阅技巧和心得。 ... [详细]
  • 在尝试使用指定的数据库进行登录时,遇到了权限问题。具体表现为用户 'lzldb'@'%' 无权访问数据库 'lzldb',错误代码为 1044 (42000)。命令行参数 `-D` 指定了数据库名称,但未能成功连接。建议检查用户的权限配置,确保其具有对目标数据库的访问权限。此外,使用密码时应谨慎处理,避免在命令行中明文显示。 ... [详细]
  • 如何在Android项目中正确导入和配置MySQL数据库驱动 ... [详细]
  • 在数据仓库管理中,定时更新程序与查询SQL之间的冲突常常导致性能瓶颈和阻塞问题。为了解决这些问题,通常需要对SQL Server进行详细的性能诊断。常用的诊断工具包括系统动态管理视图(DMVs)和扩展事件(Extended Events),这些工具能够帮助识别和分析性能瓶颈的具体原因,从而采取有效的优化措施。 ... [详细]
  • 分布式开源任务调度框架 TBSchedule 深度解析与应用实践
    本文深入解析了分布式开源任务调度框架 TBSchedule 的核心原理与应用场景,并通过实际案例详细介绍了其部署与使用方法。首先,从源码下载开始,详细阐述了 TBSchedule 的安装步骤和配置要点。接着,探讨了该框架在大规模分布式环境中的性能优化策略,以及如何通过灵活的任务调度机制提升系统效率。最后,结合具体实例,展示了 TBSchedule 在实际项目中的应用效果,为开发者提供了宝贵的实践经验。 ... [详细]
  • 在Python编程中,探讨了并发与并行的概念及其区别。并发指的是系统同时处理多个任务的能力,而并行则指在同一时间点上并行执行多个任务。文章详细解析了阻塞与非阻塞操作、同步与异步编程模型,以及IO多路复用技术的应用。通过模拟socket发送HTTP请求的过程,展示了如何创建连接、发送数据和接收响应,并强调了默认情况下socket的阻塞特性。此外,还介绍了如何利用这些技术优化网络通信性能和提高程序效率。 ... [详细]
  • 利用Java开发功能完备的电话簿应用程序,支持添加、查询与删除操作
    本研究基于Java语言开发了一款功能全面的电话簿应用程序,实现了与数据库的高效连接。该应用不仅支持添加、查询和删除联系人信息,还具备输出最大和最小ID号的功能,并能够对用户输入的ID号进行有效性验证,确保数据的准确性和完整性。详细实现方法可参阅相关文档。 ... [详细]
  • 本文深入探讨了MySQL数据库设计的核心概念与实用技巧,重点讲解了数据库设计的三范式。第一范式要求每个表都应具备主键,并确保所有字段具有原子性,即不可再进一步分割。第二范式则在此基础上进一步规范,确保所有非主键字段完全依赖于主键,避免部分依赖带来的数据冗余问题。第三范式则进一步消除传递依赖,确保非主键字段之间不存在依赖关系,从而提高数据的一致性和完整性。通过这些范式的应用,可以有效提升数据库的设计质量和性能。 ... [详细]
  • SQL Server开发技巧:修改表结构后的视图批量更新方法与实践 ... [详细]
  • Issue with the Reserved Term HOSTS in System Configuration ... [详细]
  • Spring Boot 实战(一):基础的CRUD操作详解
    在《Spring Boot 实战(一)》中,详细介绍了基础的CRUD操作,涵盖创建、读取、更新和删除等核心功能,适合初学者快速掌握Spring Boot框架的应用开发技巧。 ... [详细]
  • 深入解析:Explain命令的应用与字段详解
    深入解析:Explain命令的应用与字段详解 ... [详细]
  • 在数据库管理中,计算字段(也称为计算列)是一种重要的技术手段。计算字段通过在表定义中使用表达式或函数,自动生成并存储计算结果,从而提高查询效率和数据一致性。本文将详细介绍计算字段的创建方法、优化技巧及其在实际应用中的案例,帮助读者更好地理解和运用这一功能。 ... [详细]
author-avatar
淼淼L玖兰枢
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有