热门标签 | 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)



推荐阅读
  • PHP 编程疑难解析与知识点汇总
    本文详细解答了 PHP 编程中的常见问题,并提供了丰富的代码示例和解决方案,帮助开发者更好地理解和应用 PHP 知识。 ... [详细]
  • 深入理解 SQL 视图、存储过程与事务
    本文详细介绍了SQL中的视图、存储过程和事务的概念及应用。视图为用户提供了一种灵活的数据查询方式,存储过程则封装了复杂的SQL逻辑,而事务确保了数据库操作的完整性和一致性。 ... [详细]
  • 本文深入探讨 MyBatis 中动态 SQL 的使用方法,包括 if/where、trim 自定义字符串截取规则、choose 分支选择、封装查询和修改条件的 where/set 标签、批量处理的 foreach 标签以及内置参数和 bind 的用法。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • PHP 5.2.5 安装与配置指南
    本文详细介绍了 PHP 5.2.5 的安装和配置步骤,帮助开发者解决常见的环境配置问题,特别是上传图片时遇到的错误。通过本教程,您可以顺利搭建并优化 PHP 运行环境。 ... [详细]
  • 构建基于BERT的中文NL2SQL模型:一个简明的基准
    本文探讨了将自然语言转换为SQL语句(NL2SQL)的任务,这是人工智能领域中一项非常实用的研究方向。文章介绍了笔者在公司举办的首届中文NL2SQL挑战赛中的实践,该比赛提供了金融和通用领域的表格数据,并标注了对应的自然语言与SQL语句对,旨在训练准确的NL2SQL模型。 ... [详细]
  • MySQL中枚举类型的所有可能值获取方法
    本文介绍了一种在MySQL数据库中查询枚举(ENUM)类型字段所有可能取值的方法,帮助开发者更好地理解和利用这一数据类型。 ... [详细]
  • 在使用 DataGridView 时,如果在当前单元格中输入内容但光标未移开,点击保存按钮后,输入的内容可能无法保存。只有当光标离开单元格后,才能成功保存数据。本文将探讨如何通过调用 DataGridView 的内置方法解决此问题。 ... [详细]
  • Windows服务与数据库交互问题解析
    本文探讨了在Windows 10(64位)环境下开发的Windows服务,旨在定期向本地MS SQL Server (v.11)插入记录。尽管服务已成功安装并运行,但记录并未正确插入。我们将详细分析可能的原因及解决方案。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • 使用C#开发SQL Server存储过程的指南
    本文介绍如何利用C#在SQL Server中创建存储过程,涵盖背景、步骤和应用场景,旨在帮助开发者更好地理解和应用这一技术。 ... [详细]
  • 在当前众多持久层框架中,MyBatis(前身为iBatis)凭借其轻量级、易用性和对SQL的直接支持,成为许多开发者的首选。本文将详细探讨MyBatis的核心概念、设计理念及其优势。 ... [详细]
  • 解读MySQL查询执行计划的详细指南
    本文旨在帮助开发者和数据库管理员深入了解如何解读MySQL查询执行计划。通过详细的解析,您将掌握优化查询性能的关键技巧,了解各种访问类型和额外信息的含义。 ... [详细]
  • 利用存储过程构建年度日历表的详细指南
    本文将介绍如何使用SQL存储过程创建一个完整的年度日历表。通过实例演示,帮助读者掌握存储过程的应用技巧,并提供详细的代码解析和执行步骤。 ... [详细]
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社区 版权所有