当前位置:Gxlcms > 数据库问题 > oracle 层次化查询(生成菜单树等)

oracle 层次化查询(生成菜单树等)

时间:2021-07-01 10:21:17 帮助过:28人阅读

TABLE MENU ( "ID" NUMBER, "DATA" VARCHAR2(100), "PID" NUMBER ) insert into MENU (id, data, pid)values (7, g, 3); insert into MENU (id, data, pid)values (1, a, null); insert into MENU (id, data, pid)values (2, b, null); insert into MENU (id, data, pid)values (3, c, 2); insert into MENU (id, data, pid)values (4, d, 2); insert into MENU (id, data, pid)values (5, e, 4); insert into MENU (id, data, pid)values (6, f, 1);

技术分享

 

下面开始执行层次化查询,从PId为null的节点(该节点为根节点)开始递归查找,查找出所有的更节点下的子节点,构建出一个完整的树

select ID,DATA,nvl(TO_CHAR(PID),NULL) from menu start with PID is NULL connect by prior ID=pid

技术分享

代码解析:

(1)、start with PID is NULL  指定层次化查询的根节点,

技术分享

红框内的两个节点为根节点,并开始遍历其余的节点。

(2)、connect by prior ID=pid  当前节点的PID等于上一层节点的ID,如果满足条件,就加入到树结果集中

指定遍历查找子节点的规则----->  这一过程是递归查找,会一层一层找下去,直到不符合这一规则,则查找停止。

 

3、实现上面结果集的另一种Sql实现

select ID,DATA,nvl(TO_CHAR(PID),NULL) from menu start with (data=a or data=b) connect by prior ID=PID

技术分享

结论:根节点的定义比较灵活,但是(connect by)遍历子节点的规则,比较固定基本都是判断父节点和子节点的ID的,如果理解了这句话,层次化查询,差不多也就理解了!

 

4、  Oracle SQL 中的层次化查询会检测数据中是否存在回环(死循环),如果存在回环,则会抛出 ORA-01436: CONNECT BY loop in user data 的错误。如果在 connect by 后面加上 nocycle 则 产生回环的最后一层的节点会被删除。

 

 

如果数据中出现这种情况,产生了回环,那么在connect by 后面 加nocycle,节能

select ID,DATA,nvl(TO_CHAR(PID),NULL) from menu start with (data=a or data=b) connect by nocycle prior ID=PID 

 技术分享

just没有报错,有点郁闷,并没有删除,不知道哪里出了问题,但是至少不抱错了!!!

 

5、Oracle 还为层次化查询提供了一些伪列( Pseudo Column )。

(1)、CONNECT_BY_ISCYCLE 当这一行有一个子节点同时也是它的祖先节点时返回 1 ,否则返回 0 

(2)、CONNECT_BY_ISLEAF 当这一行是叶节点时返回 1 ,否则返回 0 。伪列 LEVEL 返回这一行在树中的层次,根为第一层。

(3)、CONNECT_BY_ROOT 查询操作符可以加在 connect by 之后的某个字段之前,表示获得这一行的根节点的该字段的值。

 

6、层次化查询还支持一个特殊的函数 SYS_CONNECT_BY_PATH , SYS_CONNECT_BY_PATH ( exp , char ),这个函数返回从根节点到这一行计算其中每个exp 表达式的值,并把它们连接成字符串,每个节点之间用 char 字符来分割。下面是一个例子。

技术分享

 

这个函数很棒,可以考虑其他的数据库也实现这个方法,这样我们处理一个树结构就很方便了!!!

 

oracle 层次化查询(生成菜单树等)

标签:表达   pid   返回   images   path   包含   connect   acl   number   

人气教程排行