oracle分层查询中的start with和connect by(树结构查询)

 ORACLE是一个关系数据库管理系统,它用表的形式组织数据,在某些表中的数据还呈现出树型 结构的联系。
 例如有如下案例:


数据为节选,字段值含义分别为税务机构代码、税务机构名称、上级税务机构代码,税务机构级别
select * from extern_dm_swjg查询的时候默认顺序就是上面的顺序,可以看出是混乱的并没有特殊结构特征。

而希望的结果如下图:

sj_swjg_dm为空即根节点的排在第一个,仔细观察 上图是树结构查询结果,可能不太直观,看下图就清楚了


1. 树结构的描述 
树结构的数据存放在表中,数据之间的层次关系即父子关系,通过表中的列与列间的关系来描述,
通过每个节点的父节点,就可以确定整个树结构。
在SELECT命令中使用CONNECT BY和START WITH 子句可以查询表中的树型结构关系。其命令格式如下:
SELECT ... 
CONNECT BY {PRIOR 列名1=列名2|列名1=PRIOR 列名2} 
[START WITH]... 
其中:CONNECT BY子句说明每行数据将是按层次顺序检索,并规定将表中的数据连入树型结构的关系中。PRIORY运算符必须放置在连接关系的两列中某一个的前面。对于节点间的父子关系,PRIOR运算符在的一侧表示父节点,在另一侧表示子节点,从而确定查找树结构是的顺序是自顶向下还是自底向上
START WITH 子句为可选项,用来标识哪个节点作为查找树型结构的根节点。若该子句被省略,则表示所有满足查询条件的行作为根节点。

例1 以树结构方式显示表的数据。
select swjg_dm,swjg_mc,sj_swjg_dm,swjg_level
from extern_dm_swjg
connect by prior swjg_dm = sj_swjg_dm
start with sj_swjg_dm is null

2. 关于PRIOR 
运算符PRIOR被放置于等号前后的位置,决定着查询时的检索顺序。

例2从节点开始自底向上查询
select swjg_dm,swjg_mc,sj_swjg_dm,swjg_level
from extern_dm_swjg
connect by  swjg_dm =  prior sj_swjg_dm
start with swjg_dm = '16107100004'

3.使用LEVEL 
    在具有树结构的表中,每一行数据都是树结构中的一个节点,由于节点所处的层次位置不同,所以每行记录都可以有一个层号。层号根据节点与根节点的距离确定。不论从哪个节点开始,该起始根节点的层号始终为1,根节点的子节点为2, 依此类推。 
    在查询中,可以使用伪列LEVEL显示每行数据的有关层次。LEVEL将返回树型结构中当前节点的层次,我们可以使用LEVEL来控制对树型结构进行遍历的深度。

例3 显示表中的各行数据及层号。
select level, swjg_dm,swjg_mc,sj_swjg_dm,swjg_level
from extern_dm_swjg
connect by prior swjg_dm = sj_swjg_dm
start with sj_swjg_dm is null

伪列LEVEL为数值型,可以在SELECT 命令中用于各种计算。 

例4 使用LEVEL改变查询结果的显示形式。
select LPAD(LEVEL,LEVEL*3,' ') as "LEVEL", swjg_dm,swjg_mc,sj_swjg_dm,swjg_level
from extern_dm_swjg
connect by prior swjg_dm = sj_swjg_dm
start with sj_swjg_dm is null 

在SELECT使用了函数LPAD,关于LPAD函数的使用,请参阅Oracal的Lpad函数
4.节点和分支的裁剪 
    在对树结构进行查询时,可以去掉表中的某些行,也可以剪掉树中的一个分支,使用WHERE子句来限定树型结构中的单个节点,以去掉树中的单个节点,但它却不影响其后代节点(自顶向下检索时)或前辈节点(自底向顶检索时)。

例5  仅剪去了树中单个节点16107100003 扶风县国家税务局
select LPAD(LEVEL,LEVEL*3,' ') as "LEVEL", swjg_dm,swjg_mc,sj_swjg_dm,swjg_level
from extern_dm_swjg
where swjg_dm !='16107100003'
connect by prior swjg_dm = sj_swjg_dm
start with sj_swjg_dm is null

 在这个查询中,仅剪去了树中单个节点swjg_dm为16107100003的,可它的子节点依然存在。若希望剪去树结构中的某个分支,
 则要用CONNECT BY 子句。CONNECT BY 子句是限定树型结构中的整个分支,既要剪除分支上的单个节点,
 也要剪除其后代节点(自顶向下检索时)或前辈节点(自底向顶检索时)。

例6  除去节点16107100003的一支
select LPAD(LEVEL,LEVEL*3,' ') as "LEVEL", swjg_dm,swjg_mc,sj_swjg_dm,swjg_level
from extern_dm_swjg
connect by prior swjg_dm = sj_swjg_dm
and swjg_dm !='16107100003'
start with sj_swjg_dm is null

    这个查询结果就与例5不同,除了剪去单个节点 16107100003 外,还将 16107100003 的子节点16107100004剪掉,即把 16107100003  这个分支剪掉了。 
    当然WHERE子句可以和CONNECT BY子句联合使用,这样能够同时剪掉单个节点和树中的某个分支。

在使用SELECT 语句来报告树结构报表时应当注意,CONNECT BY子句不能作用于出现在WHERE子句中的表连接。如果需要进行连接,可以先用树结构建立一个视图,再将这个视图与其他表连接,以完成所需要的查询。


posted on 2013-11-16 12:42  itmyhome  阅读(248)  评论(0编辑  收藏  举报

导航