Oracle 树操作(select…start with…connect by…prior)

--菜单目录结构表
create table tb_menu(
 id number(10) not null, --主键id
 title varchar2(50), --标题
 parent number(10) --parent id
)

--父菜单
insert into tb_menu_test_qbz(id, title, parent) values(1, '父菜单1',0);
insert into tb_menu_test_qbz(id, title, parent) values(2, '父菜单2',0);
insert into tb_menu_test_qbz(id, title, parent) values(3, '父菜单3',0);
insert into tb_menu_test_qbz(id, title, parent) values(4, '父菜单4',0);
insert into tb_menu_test_qbz(id, title, parent) values(5, '父菜单5',0);
--一级菜单
insert into tb_menu_test_qbz(id, title, parent) values(6, '一级菜单6',1);
insert into tb_menu_test_qbz(id, title, parent) values(7, '一级菜单7',1);
insert into tb_menu_test_qbz(id, title, parent) values(8, '一级菜单8',1);
insert into tb_menu_test_qbz(id, title, parent) values(9, '一级菜单9',2);
insert into tb_menu_test_qbz(id, title, parent) values(10, '一级菜单10',2);
insert into tb_menu_test_qbz(id, title, parent) values(11, '一级菜单11',2);
insert into tb_menu_test_qbz(id, title, parent) values(12, '一级菜单12',3);
insert into tb_menu_test_qbz(id, title, parent) values(13, '一级菜单13',3);
insert into tb_menu_test_qbz(id, title, parent) values(14, '一级菜单14',3);
insert into tb_menu_test_qbz(id, title, parent) values(15, '一级菜单15',4);
insert into tb_menu_test_qbz(id, title, parent) values(16, '一级菜单16',4);
insert into tb_menu_test_qbz(id, title, parent) values(17, '一级菜单17',4);
insert into tb_menu_test_qbz(id, title, parent) values(18, '一级菜单18',5);
insert into tb_menu_test_qbz(id, title, parent) values(19, '一级菜单19',5);
insert into tb_menu_test_qbz(id, title, parent) values(20, '一级菜单20',5);
--二级菜单
insert into tb_menu_test_qbz(id, title, parent) values(21, '二级菜单21',6);
insert into tb_menu_test_qbz(id, title, parent) values(22, '二级菜单22',6);
insert into tb_menu_test_qbz(id, title, parent) values(23, '二级菜单23',7);
insert into tb_menu_test_qbz(id, title, parent) values(24, '二级菜单24',7);
insert into tb_menu_test_qbz(id, title, parent) values(25, '二级菜单25',8);
insert into tb_menu_test_qbz(id, title, parent) values(26, '二级菜单26',9);
insert into tb_menu_test_qbz(id, title, parent) values(27, '二级菜单27',10);
insert into tb_menu_test_qbz(id, title, parent) values(28, '二级菜单28',11);
insert into tb_menu_test_qbz(id, title, parent) values(29, '二级菜单29',12);
insert into tb_menu_test_qbz(id, title, parent) values(30, '二级菜单30',13);
insert into tb_menu_test_qbz(id, title, parent) values(31, '二级菜单31',14);
insert into tb_menu_test_qbz(id, title, parent) values(32, '二级菜单32',15);
insert into tb_menu_test_qbz(id, title, parent) values(33, '二级菜单33',16);
insert into tb_menu_test_qbz(id, title, parent) values(34, '二级菜单34',17);
insert into tb_menu_test_qbz(id, title, parent) values(35, '二级菜单35',18);
insert into tb_menu_test_qbz(id, title, parent) values(36, '二级菜单36',19);
insert into tb_menu_test_qbz(id, title, parent) values(37, '二级菜单37',20);
--三级菜单
insert into tb_menu_test_qbz(id, title, parent) values(38, '三级菜单38',21);
insert into tb_menu_test_qbz(id, title, parent) values(39, '三级菜单39',22);
insert into tb_menu_test_qbz(id, title, parent) values(40, '三级菜单40',23);
insert into tb_menu_test_qbz(id, title, parent) values(41, '三级菜单41',24);
insert into tb_menu_test_qbz(id, title, parent) values(42, '三级菜单42',25);
insert into tb_menu_test_qbz(id, title, parent) values(43, '三级菜单43',26);
insert into tb_menu_test_qbz(id, title, parent) values(44, '三级菜单44',27);
insert into tb_menu_test_qbz(id, title, parent) values(45, '三级菜单45',28);
insert into tb_menu_test_qbz(id, title, parent) values(46, '三级菜单46',28);
insert into tb_menu_test_qbz(id, title, parent) values(47, '三级菜单47',29);
insert into tb_menu_test_qbz(id, title, parent) values(48, '三级菜单48',30);
insert into tb_menu_test_qbz(id, title, parent) values(49, '三级菜单49',31);
insert into tb_menu_test_qbz(id, title, parent) values(50, '三级菜单50',31);
commit;

select * from tb_menu_test_qbz;

--1)、查找树中的所有顶级父节点(辈份最长的人)。
select * from tb_menu_test_qbz m where m.parent = 0;

--2)、查找一个节点的直属子节点(所有儿子)。
select * from tb_menu_test_qbz m where m.parent=1;

--3)、查找一个节点的所有直属子节点(所有后代)。
select * from tb_menu_test_qbz m start with m.id=1 connect by m.parent=prior m.id;

--4)、查找一个节点的直属父节点(父亲)。
--c-->child, p->parent
select c.id, c.title, p.id parent_id, p.title parent_title
from tb_menu_test_qbz c, tb_menu_test_qbz p
where c.parent=p.id and c.id=6

--5)、查找一个节点的所有直属父节点(祖宗)。
select * from tb_menu_test_qbz m start with m.id=38 connect by prior m.parent=m.id;
    /*上面列出两个树型查询方式,第3条语句和第5条语句,这两条语句之间的区别在于prior关键字的位置不同,所以决定了查询的方式不同。
    当parent = prior id时,数据库会根据当前的id迭代出parent与该id相同的记录,所以查询的结果是迭代出了所有的子类记录;
    而prior parent = id时,数据库会跟据当前的parent来迭代出与当前的parent相同的id的记录,所以查询出来的结果就是所有的父类结果。*/

--6)、查询一个节点的兄弟节点(亲兄弟)。
--m.parent=m2.parent-->同一个父亲
select * from tb_menu_test_qbz m
where exists (select * from tb_menu_test_qbz m2 where m.parent=m2.parent and m2.id=6)

--7)、查询与一个节点同级的节点(族兄弟). 如果在表中设置了级别的字段,那么在做这类查询时会很轻松,同一级别的就是与那个节点同级的,在这里列出不使用该字段时的实现!
with tmp as(
      select a.*, level leaf        
      from tb_menu_test_qbz a                
      start with a.parent =0     
      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_test_qbz.*, level lev
    from tb_menu_test_qbz
    start with parent =  0
    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_test_qbz a                
      start with a.parent = 0     
      connect by a.parent = prior a.id)
select *                               
from tmp                             
where leaf = (select leaf from tmp where id = 6) - 1;

--10)、名称要列出名称全部路径。
    /*这里常见的有两种情况,一种是从顶级列出,直到当前节点的名称(或者其它属性);
    一种是从当前节点列出,直到顶级节点的名称(或其它属性)。
    举地址为例:国内的习惯是从省开始、到市、到县、到居委会的,而国外的习惯正好相反*/
--从顶部开始:
select sys_connect_by_path (title, '/')
from tb_menu_test_qbz
where id = 50
start with parent = 0
connect by parent = prior id;
--从当前节点开始:
select sys_connect_by_path (title, '/')
from tb_menu_test_qbz
start with id = 50
connect by prior parent = id;
    /*sys_connect_by_path函数就是从start with开始的地方开始遍历,并记下其遍历到的节点,
    start with开始的地方被视为根节点,将遍历到的路径根据函数中的分隔符,组成一个新的字符串,这个功能还是很强大的。*/

--11)、列出当前节点的根节点。
select connect_by_root title, tb_menu_test_qbz.*
from tb_menu_test_qbz
start with id = 50
connect by prior parent = id;

--12)、列出当前节点是否为叶子。     尤其在动态目录中,在查出的内容是否还有下级节点时,这个函数是很适用的。
select connect_by_isleaf, tb_menu_test_qbz.*
from tb_menu_test_qbz
start with parent = 0
connect by parent = prior id;
   /*connect_by_isleaf函数用来判断当前节点是否包含下级节点,如果包含的话,说明不是叶子节点,这里返回0;反之,如果不包含下级节点,这里返回1。*/

转载自: http://www.cnblogs.com/linjiqin/archive/2013/06/24/3152674.html
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值