树形查询一般用于上下级场合,使用的特殊sql语法包括level,prior,start with,connect by等,下面将就实例来说明其用法。

表定义:

create table tb_hierarchy(
id number(4,0) primary key,
name nvarchar2(20) not null,
pid number(4,0))

充值:

insert into tb_hierarchy(id,name) values('','Gates');
insert into tb_hierarchy(id,pid,name) values('','','Alice');
insert into tb_hierarchy(id,pid,name) values('','','Bill');
insert into tb_hierarchy(id,pid,name) values('','','Cindy');
insert into tb_hierarchy(id,pid,name) values('','','Douglas');
insert into tb_hierarchy(id,pid,name) values('','','Eliot');
insert into tb_hierarchy(id,pid,name) values('','','Mick');
insert into tb_hierarchy(id,pid,name) values('','','Flex');
insert into tb_hierarchy(id,pid,name) values('','','张三');
insert into tb_hierarchy(id,pid,name) values('','','李四');
insert into tb_hierarchy(id,pid,name) values('','','王五');

先让我们查出员工及其上级:

--列出员工和上级
select level,id,name,(prior name) as mngName
from tb_hierarchy
start with pid is NULL
connect by (prior id)=pid

查询结果:

SQL> select level,id,name,(prior name) as mngName
2 from tb_hierarchy
3 start with pid is NULL
4 connect by (prior id)=pid; LEVEL ID NAME MNGNAME
---------- ---------- ---------------------------------------- ----------------------------------------
1 1 Gates
2 2 Alice Gates
3 3 Bill Alice
3 4 Cindy Alice
3 5 Douglas Alice
2 6 Eliot Gates
3 7 Mick Eliot
4 9 张三 Mick
4 10 李四 Mick
4 11 王五 Mick
3 8 Flex Eliot 已选择11行。

从上面的level一列可以看出,Gates居于公司领导核心,属于董事长;他下面是alice,处于总经理地位;Alice下面有三个经理,分别是Bill,Cindy,Douglas...

这些结果是怎么查出来的呢?让我们看看SQL:

select level,id,name,(prior name) as mngName
from tb_hierarchy
start with pid is NULL
connect by (prior id)=pid

解读:

level:属于关键字,是和rownum一样的伪列,代表节点在整棵树中的层级,如Flex处于等级三,他上面有Eliot,Eliot上面有总头头Gates。

prior name:prior属于关键字,代表本条记录的上一条,如本条是(3,8,Flex);那么prior就是(2,6,Eliot);知道了prior是哪一条记录,我们就知道了prior name是Eliot,prior id就是6。

start with:这个语法告诉树形查询应该以pid是空的记录作为树的起点。

下面我们来查查以Mick为起点会是什么效果:

SQL> select level,id,name,(prior name) as mngName
2 from tb_hierarchy
3 start with name='Mick'
4 connect by (prior id)=pid; LEVEL ID NAME MNGNAME
---------- ---------- ---------------------------------------- ----------------------------------------
1 7 Mick
2 9 张三 Mick
2 10 李四 Mick
2 11 王五 Mick

结果查询出了以Mick为组长,张三李四王五为组员的苦逼外包小组。

在国企干活的人一般称底下做事的为员,管员的人为基层干部,上下都是干部的为中层干部,上面再没人的则是首长。

下面我们查查谁是员,谁是基层领导干部,谁是中层领导干部,谁是首长:

SQL> select level,id,name,(prior name) as mngName,
2 decode(level,1,1) as 首长,
3 decode(level,2,1) as 中层干部,
4 decode(level,3,1) as 基层干部,
5 decode(connect_by_isleaf,1,1) as 员工
6 from tb_hierarchy
7 start with pid is NULL
8 connect by (prior id)=pid; LEVEL ID NAME MNGNAME 首长 中层干部 基层干部 员工
---------- ---------- -------------------- -------------------- ---------- ---------- ---------- ----------
1 1 Gates 1
2 2 Alice Gates 1
3 3 Bill Alice 1 1
3 4 Cindy Alice 1 1
3 5 Douglas Alice 1 1
2 6 Eliot Gates 1
3 7 Mick Eliot 1
4 9 张三 Mick 1
4 10 李四 Mick 1
4 11 王五 Mick 1
3 8 Flex Eliot 1 1 已选择11行。

上面的语法中多了一个关键字connect_by_isleaf,它表示当前节点下面没有子节点,或是当前记录下没有地位更低的记录(996!最苦逼的一群人)

下面SQL可以把id前面加点层次:

SQL> select lpad(' ',level,' ')||id AS padid,
2 level,id,name,(prior name) as mngName,
3 decode(level,1,1) as 首长,
4 decode(level,2,1) as 中层干部,
5 decode(level,3,1) as 基层干部,
6 decode(connect_by_isleaf,1,1) as 员工
7 from tb_hierarchy
8 start with pid is NULL
9 connect by (prior id)=pid; PADID LEVEL ID NAME MNGNAME 首长 中层干部 基层干部 员工
---------- ---------- ---------- ---------- -------------------- ---------- ---------- ---------- ----------
1 1 1 Gates 1
2 2 2 Alice Gates 1
3 3 3 Bill Alice 1 1
4 3 4 Cindy Alice 1 1
5 3 5 Douglas Alice 1 1
6 2 6 Eliot Gates 1
7 3 7 Mick Eliot 1
9 4 9 张三 Mick 1
10 4 10 李四 Mick 1
11 4 11 王五 Mick 1
8 3 8 Flex Eliot 1 1 已选择11行。

下面把每个人的上级全列出来:

SQL> col path format a30;
SQL> select level,id,name,(prior name) as mngName,
2 sys_connect_by_path(name,',') as path
3 from tb_hierarchy
4 start with pid is NULL
5 connect by (prior id)=pid; LEVEL ID NAME MNGNAME PATH
---------- ---------- ---------- -------------------- ------------------------------
1 1 Gates ,Gates
2 2 Alice Gates ,Gates,Alice
3 3 Bill Alice ,Gates,Alice,Bill
3 4 Cindy Alice ,Gates,Alice,Cindy
3 5 Douglas Alice ,Gates,Alice,Douglas
2 6 Eliot Gates ,Gates,Eliot
3 7 Mick Eliot ,Gates,Eliot,Mick
4 9 张三 Mick ,Gates,Eliot,Mick,张三
4 10 李四 Mick ,Gates,Eliot,Mick,李四
4 11 王五 Mick ,Gates,Eliot,Mick,王五
3 8 Flex Eliot ,Gates,Eliot,Flex 已选择11行。

--2020年4月18日--

以上用到的全部SQL:

create table tb_hierarchy(
id number(4,0) primary key,
name nvarchar2(20) not null,
pid number(4,0)) insert into tb_hierarchy(id,name) values('','Gates');
insert into tb_hierarchy(id,pid,name) values('','','Alice');
insert into tb_hierarchy(id,pid,name) values('','','Bill');
insert into tb_hierarchy(id,pid,name) values('','','Cindy');
insert into tb_hierarchy(id,pid,name) values('','','Douglas');
insert into tb_hierarchy(id,pid,name) values('','','Eliot');
insert into tb_hierarchy(id,pid,name) values('','','Mick');
insert into tb_hierarchy(id,pid,name) values('','','Flex');
insert into tb_hierarchy(id,pid,name) values('','','张三');
insert into tb_hierarchy(id,pid,name) values('','','李四');
insert into tb_hierarchy(id,pid,name) values('','','王五'); --列出员工和上级
select level,id,name,(prior name) as mngName
from tb_hierarchy
start with pid is NULL
connect by (prior id)=pid --以mick为起点
select level,id,name,(prior name) as mngName
from tb_hierarchy
start with name='Mick'
connect by (prior id)=pid --列出是员,基层干部,中级干部和首长
select level,id,name,(prior name) as mngName,
decode(level,1,1) as 首长,
decode(level,2,1) as 中层干部,
decode(level,3,1) as 基层干部,
decode(connect_by_isleaf,1,1) as 员工
from tb_hierarchy
start with pid is NULL
connect by (prior id)=pid --加入层次列
select lpad(' ',level,' ')||id AS padid,
level,id,name,(prior name) as mngName,
decode(level,1,1) as 首长,
decode(level,2,1) as 中层干部,
decode(level,3,1) as 基层干部,
decode(connect_by_isleaf,1,1) as 员工
from tb_hierarchy
start with pid is NULL
connect by (prior id)=pid --把上级在path里全列出来
select level,id,name,(prior name) as mngName,
sys_connect_by_path(name,',') as path
from tb_hierarchy
start with pid is NULL
connect by (prior id)=pid

Oracle中树形查询使用方法的更多相关文章

  1. ORACLE跨数据库查询的方法

    原文地址:http://blog.csdn.net/huzhenwei/article/details/2533869 本文简述了通过创建database link实现Oracle跨数据库查询的方法 ...

  2. oracle 中对查询出来的数据进行切割、截取等操作

    oracle 中对查询出来的数据进行切割.截取等操作 最近遇到一个问题,需要把一个带有,的字符串拆分成多行.通过查询资料,这个操作需要使用以下2个关键知识: 1. REGEXP_SUBSTR函数 这个 ...

  3. Oracle中如何查询CLOB字段类型的内容

    注:本文来源于:<Oracle中如何查询CLOB字段类型的内容> 语法 select * from table_name where dbms_lob.instr(字段名(clod类型), ...

  4. Oracle中如何查询一个表的所有字段名和数据类型

    Oracle中如何查询一个表的所有字段名和数据类型 查询语法 select A.COLUMN_NAME,A.DATA_TYPE from user_tab_columns A where TABLE_ ...

  5. Oracle中生成uuid的方法

    Oracle中生成uuid的方法 下载LOFTER客户端 在Oracle SQL 提供了一个生成uuid的函数sys_guid: http://download.oracle.com/docs/cd/ ...

  6. Oracle中分页查询语句

    Oracle分页查询语句使我们最常用的语句之一,下面就为您介绍的Oracle分页查询语句的用法,如果您对此方面感兴趣的话,不妨一看. Oracle分页查询语句基本上可以按照本文给出的格式来进行套用.O ...

  7. Oracle 特殊字符模糊查询的方法

    最近在写DAO层的时候,遇到一个问题,就是使用like进行模糊查询时,输入下划线,无法精确查到数据,而是返回所有的数据. 这让我很好奇,百度之后才发现,原来是因为有些特殊字符需要进行转义才可以进行查询 ...

  8. oracle表空间表分区详解及oracle表分区查询使用方法(转+整理)

    欢迎和大家交流技术相关问题: 邮箱: jiangxinnju@163.com 博客园地址: http://www.cnblogs.com/jiangxinnju GitHub地址: https://g ...

  9. 【JOB】Oracle中JOB的创建方法以及一个细节的探究

    在Oracle中可以使用JOB来实现一些任务的自动化执行,类似于UNIX操作系统crontab命令的功能.简单演示一下,供参考. 1.创建表T,包含一个X字段,定义为日期类型,方便后面的定时任务测试. ...

随机推荐

  1. CSS基础知识(下)

    3.层叠 稍微复杂的样式表中都可能存在两条甚至多条规则同时选择一个元素的情况.CSS通过一种叫作层叠(cascade)的机制来处理这种冲突. 层叠机制的原理是为规则赋予不同的重要程度.最重要的是作者样 ...

  2. Docker 快速搭建 MySQL8 开发环境

    使用 Docker 快速搭建一个 MySQL8 开发环境 步骤 获取镜像 docker pull mysql:8 启动容器,密码 123456,映射 3306 端口 docker run --name ...

  3. sql server 存储过程的(包含事务)方法里面,采用游标循环,批量删除(修改)数据

    sqlserver 数据库 1.下面是完整的 在存储过程中 使用游标进行 循环删除的实例(包括存储过程中,事务的应用) 2.有问题的话,欢迎随时讨饶我,相信大家看下注释应该就能明白了,很简单的一个,小 ...

  4. java_String类、StringBuilder类、Arrays类、Math类的使用

    String类 java.lang.String 类代表字符串.Java程序中所有的字符串文字(例如 “abc” )都可以被看作是实现此类的实例 构造方法 java.lang.String :此类不需 ...

  5. 2020-03-25:快排、堆排和归并都是O(nlog n)的算法,为何JDK选择快速排序?

    福哥答案2020-03-26: 口诀如下:冒选插希快 堆归计桶基(冒泡,选择,插入,希尔,快速,堆,归并,计数,桶,基数)冒线 平平 稳常小选平 平平 不常小插线 平平 稳常序希线 四组 不常组快四 ...

  6. C#LeetCode刷题之#434-字符串中的单词数​​​​​​​(Number of Segments in a String)

    问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/3941 访问. 统计字符串中的单词个数,这里的单词指的是连续的不是 ...

  7. MyBatis-Pro,新一代的MyBatis增强框架

    框架功能 内置提供基础CRUD方法 提供根据方法名自进行单表查询(包括查询.统计.删除等) 接入方法 Spring Boot <dependency> <groupId>com ...

  8. css如何让子元素在父元素中水平垂直居中

    方法一: display:flex <!DOCTYPE html> <html lang="en"> <head> <meta chars ...

  9. JavaScript基础-06-正则表达式

    正则表达式 1. 正则表达式用于定义一些字符串的规则:计算机可以根据正则表达式,来检查一个字符串是否符合规则,将字符串中符合规则的内容提取出来. 2. 创建正则表达式对象: var reg=new R ...

  10. 浏览器自动化的一些体会5 webBrowser控件之winform和webBrowser的交互

    从winform访问webBrowser,大致就是利用webBrowser提供的解析dom的方法以及用InvokeScript方法执行javascript.这个相对比较简单. 从webBrowser访 ...