系列文章目录

【SQL开发实战技巧】系列(一):关于SQL不得不说的那些事
【SQL开发实战技巧】系列(二):简单单表查询
【SQL开发实战技巧】系列(三):SQL排序的那些事
【SQL开发实战技巧】系列(四):从执行计划讨论UNION ALL与空字符串&UNION与OR的使用注意事项
【SQL开发实战技巧】系列(五):从执行计划看IN、EXISTS 和 INNER JOIN效率,我们要分场景不要死记网上结论
【SQL开发实战技巧】系列(六):从执行计划看NOT IN、NOT EXISTS 和 LEFT JOIN效率,记住内外关联条件不要乱放
【SQL开发实战技巧】系列(七):从有重复数据前提下如何比较出两个表中的差异数据及对应条数聊起
【SQL开发实战技巧】系列(八):聊聊如何插入数据时比约束更灵活的限制数据插入以及怎么一个insert语句同时插入多张表
【SQL开发实战技巧】系列(九):一个update误把其他列数据更新成空了?Merge改写update!给你五种删除重复数据的写法!
【SQL开发实战技巧】系列(十):从拆分字符串、替换字符串以及统计字符串出现次数说起
【SQL开发实战技巧】系列(十一):拿几个案例讲讲translate|regexp_replace|listagg|wmsys.wm_concat|substr|regexp_substr常用函数
【SQL开发实战技巧】系列(十二):三问(如何对字符串字母去重后按字母顺序排列字符串?如何识别哪些字符串中包含数字?如何将分隔数据转换为多值IN列表?)
【SQL开发实战技巧】系列(十三):讨论一下常用聚集函数&通过执行计划看sum()over()对员工工资进行累加
【SQL开发实战技巧】系列(十四):计算消费后的余额&计算银行流水累计和&计算各部门工资排名前三位的员工
【SQL开发实战技巧】系列(十五):查找最值所在行数据信息及快速计算总和百之max/min() keep() over()、fisrt_value、last_value、ratio_to_report
【SQL开发实战技巧】系列(十六):数据仓库中时间类型操作(初级)日、月、年、时、分、秒之差及时间间隔计算
【SQL开发实战技巧】系列(十七):数据仓库中时间类型操作(初级)确定两个日期之间的工作天数、计算—年中周内各日期出现次数、确定当前记录和下一条记录之间相差的天数
【SQL开发实战技巧】系列(十八):数据仓库中时间类型操作(进阶)INTERVAL、EXTRACT以及如何确定一年是否为闰年及周的计算
【SQL开发实战技巧】系列(十九):数据仓库中时间类型操作(进阶)如何一个SQL打印当月或一年的日历?如何确定某月内第一个和最后—个周内某天的日期?
【SQL开发实战技巧】系列(二十):数据仓库中时间类型操作(进阶)获取季度开始结束时间以及如何统计非连续性时间的数据
【SQL开发实战技巧】系列(二十一):数据仓库中时间类型操作(进阶)识别重叠的日期范围,按指定10分钟时间间隔汇总数据
【SQL开发实战技巧】系列(二十二):数仓报表场景☞ 从分析函数效率一定快吗聊一聊结果集分页和隔行抽样实现方式
【SQL开发实战技巧】系列(二十三):数仓报表场景☞ 如何对数据排列组合去重以及通过如何找到包含最大值和最小值的记录这个问题再次用执行计划给你证明分析函数性能不一定高
【SQL开发实战技巧】系列(二十四):数仓报表场景☞通过案例执行计划详解”行转列”,”列转行”是如何实现的
【SQL开发实战技巧】系列(二十五):数仓报表场景☞结果集中的重复数据只显示一次以及计算部门薪资差异高效的写法以及如何对数据进行快速分组
【SQL开发实战技巧】系列(二十六):数仓报表场景☞聊聊ROLLUP、UNION ALL是如何分别做分组合计的以及如何识别哪些行是做汇总的结果行
【SQL开发实战技巧】系列(二十七):数仓报表场景☞通过对移动范围进行聚集来详解分析函数开窗原理以及如何一个SQL打印九九乘法表
【SQL开发实战技巧】系列(二十八):数仓报表场景☞人员分布问题以及不同组(分区)同时聚集如何实现
【SQL开发实战技巧】系列(二十九):数仓报表场景☞简单的树形(分层)查询以及如何确定根节点、分支节点和叶子节点
【SQL开发实战技巧】系列(三十):数仓报表场景☞树形(分层)查询如何排序?以及如何在树形查询中正确的使用where条件
【SQL开发实战技巧】系列(三十一):数仓报表场景☞分层查询如何只查询树形结构某一个分支?如何剪掉一个分支?


文章目录

  • 系列文章目录
  • 前言
  • 一、如何只查询树形的一个分支
  • 二、如何正确的剪去一个分支
  • 总结

前言

本篇文章讲解的主要内容是:第一个案例给大家介绍在树形(分层)查询中,我们如何只查询出来树形的一个分支?如何剪去分支,不能在WHERE中加条件。
【SQL开发实战技巧】这一系列博主当作复习旧知识来进行写作,毕竟SQL开发在数据分析场景非常重要且基础,面试也会经常问SQL开发和调优经验,相信当我写完这一系列文章,也能再有所收获,未来面对SQL面试也能游刃有余~。


一、如何只查询树形的一个分支

现在有个需求:在分层查询中,我们如何只查询出来树形的一个分支?
说到这里,你会不会脑海中直接想where了?
可惜的是,我们不能用where过滤来做,我们先说一下正确的实现需求的写法:
其实我们用START WITH指定分支的起点就可以啦,看下面查询:

SELECT lpad('->', (level - 1) * 2, '->') || empno AS 员工编码, ename AS 姓名, mgr AS 主管编码, (PRIOR ename) AS 主管姓名, level as 级别, decode(level, 1, 1) as 根节点, decode(connect_by_isleaf, 1, 1) as 叶子节点, CASE WHEN (connect_by_isleaf = 0 AND LEVEL > 1) THEN1 END AS 分支节点, deptno as 部门编码, sys_connect_by_path(ename, '->') as enamesFROM emp START WITH empno=7566CONNECT BY (PRIOR empno) = mgr order siblings by empno;员工编码 姓名主管编码 主管姓名 级别根节点 叶子节点 分支节点 部门编码 ENAMES-------------------------------------------------------------------------------- ---------- ----- ---------- ---------- ---------- ---------- ---------- ---- --------------------------------------------------------------------------------7566 JONES 7839 11 20 ->JONES->7788 SCOTT 7566 JONES 21 20 ->JONES->SCOTT->->7876 ADAMS 7788 SCOTT 3 120 ->JONES->SCOTT->ADAMS->7902 FORD7566 JONES 21 20 ->JONES->FORD->->7369 SMITH 7902 FORD3 120 ->JONES->FORD->SMITH

说完了正确的写法,这里给大家说一下为什么不能用where过滤来实现这个需求:
我们用前面的案例说一下,看下面查询:

SELECT lpad('->', (level - 1) * 2, '->') || empno AS 员工编码, ename AS 姓名, mgr AS 主管编码, (PRIOR ename) AS 主管姓名, level as 级别, decode(level, 1, 1) as 根节点, decode(connect_by_isleaf, 1, 1) as 叶子节点, CASE WHEN (connect_by_isleaf = 0 AND LEVEL > 1) THEN1 END AS 分支节点, deptno as 部门编码, sys_connect_by_path(ename, '->') as enamesFROM emp START WITH mgr is nullCONNECT BY (PRIOR empno) = mgr order siblings by empno;员工编码 姓名主管编码 主管姓名 级别根节点 叶子节点 分支节点 部门编码 ENAMES-------------------------------------------------------------------------------- ---------- ----- ---------- ---------- ---------- ---------- ---------- ---- --------------------------------------------------------------------------------7839 KING 11 10 ->KING->7566 JONES 7839 KING21 20 ->KING->JONES->->7788 SCOTT 7566 JONES 31 20 ->KING->JONES->SCOTT->->->7876 ADAMS 7788 SCOTT 4 120 ->KING->JONES->SCOTT->ADAMS->->7902 FORD7566 JONES 31 20 ->KING->JONES->FORD->->->7369 SMITH 7902 FORD4 120 ->KING->JONES->FORD->SMITH->7698 BLAKE 7839 KING21 30 ->KING->BLAKE->->7499 ALLEN 7698 BLAKE 3 130 ->KING->BLAKE->ALLEN->->7521 WARD7698 BLAKE 3 130 ->KING->BLAKE->WARD->->7654 MARTIN7698 BLAKE 3 130 ->KING->BLAKE->MARTIN->->7844 TURNER7698 BLAKE 3 130 ->KING->BLAKE->TURNER->->7900 JAMES 7698 BLAKE 3 130 ->KING->BLAKE->JAMES->7782 CLARK 7839 KING21 10 ->KING->CLARK->->7934 MILLER7782 CLARK 3 110 ->KING->CLARK->MILLER14 rows selected

分析上面查询结果,如果我只想要mrg=7566的这个分支结构的数据,用where条件的话,我们上一篇文章说过:“如果你有这种过滤+树形查询的需求,一定一定要先过滤,用子查询嵌套一次,再进行树形查询”
那么接下来我们这么查询:

SELECT lpad('->', (level - 1) * 2, '->') || empno AS 员工编码, ename AS 姓名, mgr AS 主管编码, (PRIOR ename) AS 主管姓名, level as 级别, decode(level, 1, 1) as 根节点, decode(connect_by_isleaf, 1, 1) as 叶子节点, CASE WHEN (connect_by_isleaf = 0 AND LEVEL > 1) THEN1 END AS 分支节点, deptno as 部门编码, sys_connect_by_path(ename, '->') as enamesFROM (select * fromemp where mgr=7566) START WITH empno=7566CONNECT BY (PRIOR empno) = mgr order siblings by empno;员工编码 姓名主管编码 主管姓名 级别根节点 叶子节点 分支节点 部门编码 ENAMES-------------------------------------------------------------------------------- ---------- ----- ---------- ---------- ---------- ---------- ---------- ---- --------------------------------------------------------------------------------SQL> 

诶嘿,可以看到哈,没结果!!!没查出来任何信息!为什么?因为子查询select * from emp where mgr=7566把根节点empno=7566给过滤掉了,看下子查询结果:

SQL> select * fromemp where mgr=75662;EMPNO ENAMEJOB MGR HIREDATESALCOMM DEPTNO----- ---------- --------- ----- ----------- --------- --------- ------ 7788 SCOTTANALYST7566 1987-4-19 3000.00 20 7902 FORD ANALYST7566 1981-12-3 3000.00 20SQL> 

那会不会有人这么想?我把这个根节点empno=7566加上不就得了?如下查询:

SQL> SELECT lpad('->', (level - 1) * 2, '->') || empno AS 员工编码,2 ename AS 姓名,3 mgr AS 主管编码,4 (PRIOR ename) AS 主管姓名,5 level as 级别,6 decode(level, 1, 1) as 根节点,7 decode(connect_by_isleaf, 1, 1) as 叶子节点,8 CASE9 WHEN (connect_by_isleaf = 0 AND LEVEL > 1) THEN 101 11 END AS 分支节点, 12 deptno as 部门编码, 13 sys_connect_by_path(ename, '->') as enames 14FROM (select * fromemp where mgr=7566 union all select * from emp where empno=7566) 15 START WITH empno=7566 16CONNECT BY (PRIOR empno) = mgr 17 order siblings by empno;员工编码 姓名主管编码 主管姓名 级别根节点 叶子节点 分支节点 部门编码 ENAMES-------------------------------------------------------------------------------- ---------- ----- ---------- ---------- ---------- ---------- ---------- ---- --------------------------------------------------------------------------------7566 JONES 7839 11 20 ->JONES->7788 SCOTT 7566 JONES 2 120 ->JONES->SCOTT->7902 FORD7566 JONES 2 120 ->JONES->FORD

那么问题来了,(7788,7902)从第一个查询结果看它俩是分支节点呀,他们下面分别有7876,7369呢,而且我是在本节开头的时候给了你确定的答案,揣着答案找肯定明白,但是当你不知道正确答案时候呢?所以说,用where是不行滴!

二、如何正确的剪去一个分支

接上面的案例:

SELECT lpad('->', (level - 1) * 2, '->') || empno AS 员工编码, ename AS 姓名, mgr AS 主管编码, (PRIOR ename) AS 主管姓名, level as 级别, decode(level, 1, 1) as 根节点, decode(connect_by_isleaf, 1, 1) as 叶子节点, CASE WHEN (connect_by_isleaf = 0 AND LEVEL > 1) THEN1 END AS 分支节点, deptno as 部门编码, sys_connect_by_path(ename, '->') as enamesFROM emp START WITH mgr is nullCONNECT BY (PRIOR empno) = mgr order siblings by empno;员工编码 姓名主管编码 主管姓名 级别根节点 叶子节点 分支节点 部门编码 ENAMES-------------------------------------------------------------------------------- ---------- ----- ---------- ---------- ---------- ---------- ---------- ---- --------------------------------------------------------------------------------7839 KING 11 10 ->KING->7566 JONES 7839 KING21 20 ->KING->JONES->->7788 SCOTT 7566 JONES 31 20 ->KING->JONES->SCOTT->->->7876 ADAMS 7788 SCOTT 4 120 ->KING->JONES->SCOTT->ADAMS->->7902 FORD7566 JONES 31 20 ->KING->JONES->FORD->->->7369 SMITH 7902 FORD4 120 ->KING->JONES->FORD->SMITH->7698 BLAKE 7839 KING21 30 ->KING->BLAKE->->7499 ALLEN 7698 BLAKE 3 130 ->KING->BLAKE->ALLEN->->7521 WARD7698 BLAKE 3 130 ->KING->BLAKE->WARD->->7654 MARTIN7698 BLAKE 3 130 ->KING->BLAKE->MARTIN->->7844 TURNER7698 BLAKE 3 130 ->KING->BLAKE->TURNER->->7900 JAMES 7698 BLAKE 3 130 ->KING->BLAKE->JAMES->7782 CLARK 7839 KING21 10 ->KING->CLARK->->7934 MILLER7782 CLARK 3 110 ->KING->CLARK->MILLER14 rows selected

现在有个新的需求:要求剪去7698开始的这个分支。
同样,剪去分支也不能在WHERE中加条件,因为树形查询递归是根据条件(PRIOR empno)=mgr进行的,所以在下列语句加条件就可以。

SELECT lpad('->', (level - 1) * 2, '->') || empno AS 员工编码, ename AS 姓名, mgr AS 主管编码, (PRIOR ename) AS 主管姓名, level as 级别, decode(level, 1, 1) as 根节点, decode(connect_by_isleaf, 1, 1) as 叶子节点, CASE WHEN (connect_by_isleaf = 0 AND LEVEL > 1) THEN1 END AS 分支节点, deptno as 部门编码, sys_connect_by_path(ename, '->') as enamesFROM emp START WITH mgr is nullCONNECT BY (PRIOR empno) = mgrandempno !=7698 order siblings by empno;员工编码 姓名主管编码 主管姓名 级别根节点 叶子节点 分支节点 部门编码 ENAMES-------------------------------------------------------------------------------- ---------- ----- ---------- ---------- ---------- ---------- ---------- ---- --------------------------------------------------------------------------------7839 KING 11 10 ->KING->7566 JONES 7839 KING21 20 ->KING->JONES->->7788 SCOTT 7566 JONES 31 20 ->KING->JONES->SCOTT->->->7876 ADAMS 7788 SCOTT 4 120 ->KING->JONES->SCOTT->ADAMS->->7902 FORD7566 JONES 31 20 ->KING->JONES->FORD->->->7369 SMITH 7902 FORD4 120 ->KING->JONES->FORD->SMITH->7782 CLARK 7839 KING21 10 ->KING->CLARK->->7934 MILLER7782 CLARK 3 110 ->KING->CLARK->MILLER8 rows selected

总结

本篇文章讲解的主要内容是:第一个案例给大家介绍在树形(分层)查询中,我们如何只查询出来树形的一个分支?如何剪去分支,不能在WHERE中加条件。