`
sdfiyon
  • 浏览: 21514 次
  • 性别: Icon_minigender_1
  • 来自: 济南
社区版块
存档分类
最新评论

Oracle 树操作

 
阅读更多

oracle树查询的最重要的就是select…start with…connect by…prior语法了。依托于该语法,我们可以将一个表形结构的以树的顺序列出来。在下面列述了oracle中树型查询的常用查询方式以及经常使用的与树查询相关的oracle特性函数等,在这里只涉及到一张表中的树查询方式而不涉及多表中的关联等。

1、准备测试表和测试数据

--菜单目录结构表

create table tb_menu(

   id     number(10) not null, --主键id

   title  varchar2(50), --标题

   parent number(10) --parent id

)

 

--父菜单

insert into tb_menu(id, title, parent) values(1, '父菜单1',null);

insert into tb_menu(id, title, parent) values(2, '父菜单2',null);

insert into tb_menu(id, title, parent) values(3, '父菜单3',null);

insert into tb_menu(id, title, parent) values(4, '父菜单4',null);

insert into tb_menu(id, title, parent) values(5, '父菜单5',null);

--一级菜单

insert into tb_menu(id, title, parent) values(6, '一级菜单6',1);

insert into tb_menu(id, title, parent) values(7, '一级菜单7',1);

insert into tb_menu(id, title, parent) values(8, '一级菜单8',1);

insert into tb_menu(id, title, parent) values(9, '一级菜单9',2);

insert into tb_menu(id, title, parent) values(10, '一级菜单10',2);

insert into tb_menu(id, title, parent) values(11, '一级菜单11',2);

insert into tb_menu(id, title, parent) values(12, '一级菜单12',3);

insert into tb_menu(id, title, parent) values(13, '一级菜单13',3);

insert into tb_menu(id, title, parent) values(14, '一级菜单14',3);

insert into tb_menu(id, title, parent) values(15, '一级菜单15',4);

insert into tb_menu(id, title, parent) values(16, '一级菜单16',4);

insert into tb_menu(id, title, parent) values(17, '一级菜单17',4);

insert into tb_menu(id, title, parent) values(18, '一级菜单18',5);

insert into tb_menu(id, title, parent) values(19, '一级菜单19',5);

insert into tb_menu(id, title, parent) values(20, '一级菜单20',5);

--二级菜单

insert into tb_menu(id, title, parent) values(21, '二级菜单21',6);

insert into tb_menu(id, title, parent) values(22, '二级菜单22',6);

insert into tb_menu(id, title, parent) values(23, '二级菜单23',7);

insert into tb_menu(id, title, parent) values(24, '二级菜单24',7);

insert into tb_menu(id, title, parent) values(25, '二级菜单25',8);

insert into tb_menu(id, title, parent) values(26, '二级菜单26',9);

insert into tb_menu(id, title, parent) values(27, '二级菜单27',10);

insert into tb_menu(id, title, parent) values(28, '二级菜单28',11);

insert into tb_menu(id, title, parent) values(29, '二级菜单29',12);

insert into tb_menu(id, title, parent) values(30, '二级菜单30',13);

insert into tb_menu(id, title, parent) values(31, '二级菜单31',14);

insert into tb_menu(id, title, parent) values(32, '二级菜单32',15);

insert into tb_menu(id, title, parent) values(33, '二级菜单33',16);

insert into tb_menu(id, title, parent) values(34, '二级菜单34',17);

insert into tb_menu(id, title, parent) values(35, '二级菜单35',18);

insert into tb_menu(id, title, parent) values(36, '二级菜单36',19);

insert into tb_menu(id, title, parent) values(37, '二级菜单37',20);

--三级菜单

insert into tb_menu(id, title, parent) values(38, '三级菜单38',21);

insert into tb_menu(id, title, parent) values(39, '三级菜单39',22);

insert into tb_menu(id, title, parent) values(40, '三级菜单40',23);

insert into tb_menu(id, title, parent) values(41, '三级菜单41',24);

insert into tb_menu(id, title, parent) values(42, '三级菜单42',25);

insert into tb_menu(id, title, parent) values(43, '三级菜单43',26);

insert into tb_menu(id, title, parent) values(44, '三级菜单44',27);

insert into tb_menu(id, title, parent) values(45, '三级菜单45',28);

insert into tb_menu(id, title, parent) values(46, '三级菜单46',28);

insert into tb_menu(id, title, parent) values(47, '三级菜单47',29);

insert into tb_menu(id, title, parent) values(48, '三级菜单48',30);

insert into tb_menu(id, title, parent) values(49, '三级菜单49',31);

insert into tb_menu(id, title, parent) values(50, '三级菜单50',31);

commit;

 

select * from tb_menu;

 

 

parent字段存储的是上级id,如果是顶级父节点,该parent为null(得补充一句,当初的确是这样设计的,不过现在知道,表中最好别有null记录,这会引起全文扫描,建议改成0代替)。

2、树操作
我们从最基本的操作,逐步列出树查询中常见的操作,所有查询出来的节点以家族中的辈份作比方。

1)、查找树中的所有顶级父节点(辈份最长的人)。 假设这个树是个目录结构,那么第一个操作总是找出所有的顶级节点,再根据该节点找到其下属节点。

select * from tb_menu m where m.parent is null;

2)、查找一个节点的直属子节点(所有儿子)。 如果查找的是直属子类节点,也是不用用到树型查询的。

 

select * from tb_menu m where m.parent=1;

 3)、查找一个节点的所有直属子节点(所有后代)。

 

select * from tb_menu m start with m.id=1 connect by m.parent=prior m.id;

 这个查找的是id为1的节点下的所有直属子类节点,包括子辈的和孙子辈的所有直属节点。

 

4)、查找一个节点的直属父节点(父亲)。 如果查找的是节点的直属父节点,也是不用用到树型查询的。

--c-->child, p->parent

select c.id, c.title, p.id parent_id, p.title parent_title

from tb_menu c, tb_menu p

where c.parent=p.id and c.id=6

 5)、查找一个节点的所有直属父节点(祖宗)。

 

select * from tb_menu m start with m.id=38 connect by prior m.parent=m.id;

 这里查找的就是id为1的所有直属父节点,打个比方就是找到一个人的父亲、祖父等。但是值得注意的是这个查询出来的结果的顺序是先列出子类节点再列出父类节点,姑且认为是个倒序吧。

 

上面列出两个树型查询方式,第3条语句和第5条语句,这两条语句之间的区别在于prior关键字的位置不同,所以决定了查询的方式不同。 当parent = prior id时,数据库会根据当前的id迭代出parent与该id相同的记录,所以查询的结果是迭代出了所有的子类记录;而prior parent = id时,数据库会跟据当前的parent来迭代出与当前的parent相同的id的记录,所以查询出来的结果就是所有的父类结果。

以下是一系列针对树结构的更深层次的查询,这里的查询不一定是最优的查询方式,或许只是其中的一种实现而已。

6)、查询一个节点的兄弟节点(亲兄弟)。

--m.parent=m2.parent-->同一个父亲

select * from tb_menu m

where exists (select * from tb_menu m2 where m.parent=m2.parent and m2.id=6)

 7)、查询与一个节点同级的节点(族兄弟)。 如果在表中设置了级别的字段,那么在做这类查询时会很轻松,同一级别的就是与那个节点同级的,在这里列出不使用该字段时的实现!

 

with tmp as(

      select a.*, level leaf        

      from tb_menu a                

      start with a.parent is null     

      connect by a.parent = prior a.id)

select *                               

from tmp                             

where leaf = (select leaf from tmp where id = 50);

 这里使用两个技巧,一个是使用了level来标识每个节点在表中的级别,还有就是使用with语法模拟出了一张带有级别的临时表。

 

8)、查询一个节点的父节点的的兄弟节点(伯父与叔父)。  

 

with tmp as(

    select tb_menu.*, level lev

    from tb_menu

    start with parent is null

    connect by parent = prior id)

    

select b.*

from tmp b,(select *

            from tmp

            where id = 21 and lev = 2) a

where b.lev = 1

 

union all

 

select *

from tmp

where parent = (select distinct x.id

                from tmp x, --祖父

                     tmp y, --父亲

                     (select *

                      from tmp

                      where id = 21 and lev > 2) z --儿子

                where y.id = z.parent and x.id = y.parent);

 

 

这里查询分成以下几步。
首先,将第7个一样,将全表都使用临时表加上级别;
其次,根据级别来判断有几种类型,以上文中举的例子来说,有三种情况:
(1)当前节点为顶级节点,即查询出来的lev值为1,那么它没有上级节点,不予考虑。
(2)当前节点为2级节点,查询出来的lev值为2,那么就只要保证lev级别为1的就是其上级节点的兄弟节点。
(3)其它情况就是3以及以上级别,那么就要选查询出来其上级的上级节点(祖父),再来判断祖父的下级节点都是属于该节点的上级节点的兄弟节点。
最后,就是使用union将查询出来的结果进行结合起来,形成结果集。

9)、查询一个节点的父节点的同级节点(族叔)。
这个其实跟第7种情况是相同的。

with tmp as(

      select a.*, level leaf        

      from tb_menu a                

      start with a.parent is null     

      connect by a.parent = prior a.id)

select *                               

from tmp                             

where leaf = (select leaf from tmp where id = 6) - 1;

 基本上,常见的查询在里面了,不常见的也有部分了。其中,查询的内容都是节点的基本信息,都是数据表中的基本字段,但是在树查询中还有些特殊需求,是对查询数据进行了处理的,常见的包括列出树路径等。

 

补充一个概念,对于数据库来说,根节点并不一定是在数据库中设计的顶级节点,对于数据库来说,根节点就是start with开始的地方。

下面列出的是一些与树相关的特殊需求。

10)、名称要列出名称全部路径。
这里常见的有两种情况,一种是从顶级列出,直到当前节点的名称(或者其它属性);一种是从当前节点列出,直到顶级节点的名称(或其它属性)。举地址为例:国内的习惯是从省开始、到市、到县、到居委会的,而国外的习惯正好相反(老师说的,还没接过国外的邮件,谁能寄个瞅瞅  )。
从顶部开始:

select sys_connect_by_path (title, '/')

from tb_menu

where id = 50

start with parent is null

connect by parent = prior id;

 从当前节点开始:

 

select sys_connect_by_path (title, '/')

from tb_menu

start with id = 50

connect by prior parent = id;

 在这里我又不得不放个牢骚了。oracle只提供了一个sys_connect_by_path函数,却忘了字符串的连接的顺序。在上面的例子中,第一个sql是从根节点开始遍历,而第二个sql是直接找到当前节点,从效率上来说已经是千差万别,更关键的是第一个sql只能选择一个节点,而第二个sql却是遍历出了一颗树来。再次ps一下。

 

sys_connect_by_path函数就是从start with开始的地方开始遍历,并记下其遍历到的节点,start with开始的地方被视为根节点,将遍历到的路径根据函数中的分隔符,组成一个新的字符串,这个功能还是很强大的。

11)、列出当前节点的根节点。
在前面说过,根节点就是start with开始的地方。

select connect_by_root title, tb_menu.*

from tb_menu

start with id = 50

connect by prior parent = id;

 connect_by_root函数用来列的前面,记录的是当前节点的根节点的内容。

 

12)、列出当前节点是否为叶子。
这个比较常见,尤其在动态目录中,在查出的内容是否还有下级节点时,这个函数是很适用的。

select connect_by_isleaf, tb_menu.*

from tb_menu

start with parent is null

connect by parent = prior id;

 connect_by_isleaf函数用来判断当前节点是否包含下级节点,如果包含的话,说明不是叶子节点,这里返回0;反之,如果不包含下级节点,这里返回1。

 

至此,oracle树型查询基本上讲完了,以上的例子中的数据是使用到做过的项目中的数据,因为里面的内容可能不好理解,所以就全部用一些新的例子来进行阐述。以上所有sql都在本机上测试通过,也都能实现相应的功能,但是并不能保证是解决这类问题的最优方案(如第8条明显写成存储过程会更好).

分享到:
评论

相关推荐

    Oracle 树操作(SQL查询成树菜单)

    Oracle 树操作(SQL查询成树菜单)

    Oracle 树操作(SQL查询成树状菜单)

    Oracle中树的操作

    Oracel中如何将一个树状节点复制成一个新的树节点, 使用PL/SQL表,巧妙实现树中整个节点的插入业务

    Oracle数据库实验操作

    实验29:树结构的查询start with子句 66 实验30:高级dml操作 68 第二部分pl/sql基础 69 匿名块的编写 69 实验31:书写一个最简单的块,运行并查看结果 69 实验32:在块中操作变量 70 实验33:在块中操作表的数据 71...

    树控件连接oracle数据库,包括节点编辑、删除、弹出菜单等

    VS2008实现了树控件与oracle数据库的连接与节点的操作等功能。

    oracle10g课堂练习I(1)

    B 树索引 7-28 位图索引 7-30 索引选项 7-32 创建索引 7-34 什么是视图 7-35 创建视图 7-36 序列 7-37 创建序列 7-38 使用序列 7-40 临时表 7-41 临时表:注意事项 7-43 数据字典:概览 7-44 数据字典...

    delphi 树控件使用连接Oracle数据库代码

    delphi根据orale数据表建立树控件子目录,并根据子目录编号选择不同的SQL语言,操作Oracle数据库

    Oracle DBA workshop1 (中文版)

    前言 1 简介 课程目标1-2 建议日程表1-3 课程目标1-4 Oracle 产品和服务1-5 Oracle Database 10g:“g...B 树索引7-28 位图索引7-30 索引选项7-32 创建索引7-34 vi 什么是视图7-35 创建视图7-36 序列7-37 创建序列7-38

    ORACLE9i_优化设计与系统调整

    §13.2.3 使用B树索引- 166 §13.2.4 使用反向键索引- 166 §13.2.5 使用索引组织表 166 §13.3 使用范围索引 166 §13.4 使用簇 - 167 §13.5 使用Hash 簇 - 168 §13.5.1 何时创建Hash 簇 168 §13.5.2 创建Hash ...

    Oracle自学(学习)材料 (共18章 偏理论一点)

    12 管理索引 目标 12-2 索引的分类 12-3 B 树索引 12-4 位图索引 12-6 B 树索引和位图索引的比较 12-7 创建普通 B 树索引 12-8 创建索引:指导 12-10 创建位图索引 12-11 修改索引的储存参数 12-12 分配和回收索引...

    深入浅出oracle

    循序渐进地学习使用Oracle开发所需知识 n 第一部分:关系数据库基础 数据库是什么、有什么、能干什么;...行列转换、分页、树 n 第四部分:JDBC 使用Java操作数据库 《深入浅出学Oracle》——系列精品教程

    oracle学习文档 笔记 全面 深刻 详细 通俗易懂 doc word格式 清晰 连接字符串

    1. 层次结构模型: 层次结构模型实质上是一种有根结点的定向有序树,IMS(Information Manage-mentSystem)是其典型代表。 2. 网状结构模型:按照网状数据结构建立的数据库系统称为网状数据库系统,其典型代表是DBTG...

    Oracle Database 11g初学者指南--详细书签版

    6.4 Oracle Database 11g的操作模式 160 6.4.1 操作模式 160 6.4.2 数据库和实例关闭 161 6.5 OEM使用初步 162 6.5.1 实例配置 163 6.5.2 用户会话 163 6.5.3 资源消费者组 163 6.5.4 模式、安全和存储管理 ...

    ORACLE重建索引总结

    最后,一旦操作成功,删除原有索引树,降临时数据段重命名为新的索引。 需要注意的是alter index index_name rebuild 命令中必须使用tablespace字句,以保证重建工作是在现有索引相同的表空间进行。 2.3、...

    Oracle 10g 学习笔记

    这个笔记是学习oracle数据库过程中整理出来的,比较详细,适合于初学者。 │ oracle与tomcat端口冲突.txt │ Oracle学习笔记.pdf │ Oracle学习笔记.wps │ 安装Oracle后myEclipse不能正常使用.txt │ 手工配置...

    Oracle SQL高级编程(资深Oracle专家力作,OakTable团队推荐)--随书源代码

    3.1.1 如何选择全扫描操作 56 3.1.2 全扫描与舍弃 59 3.1.3 全扫描与多块读取 60 3.1.4 全扫描与高水位线 60 3.2 索引扫描访问方法 65 3.2.1 索引结构 66 3.2.2 索引扫描类型 68 3.2.3 索引唯一扫描 71 ...

    Oracle优化日记:一个金牌DBA的故事 白鳝.扫描版

    STATSPACK报告分析之DB Cache1优化小技巧 STATSPACK报告分析之PGA6月15日 青岛今日点评优化小技巧 如何 优化闩锁6月16日 SQL 优化效果今日点评优化小技巧 如何转储数据块优化小技巧 通过树转储来分析索引碎片6月17日...

    ORACLE教材

    B树索引(默认) 位图索引 函数索引 视图 序列 利用OEM操作 第九章:备份与恢复 脱机备份与恢复 联机备份与恢复 逻辑备份与恢复 第十章:sqlplus基础 设置SQL*PLUS的运行环境 格式化查询命令 第十一...

Global site tag (gtag.js) - Google Analytics