oracle的connect by用法

这篇博客详细介绍了Oracle数据库中的CONNECT BY操作,包括如何使用START WITH指定起始节点,利用CONNECT BY定义父子行关系,以及PRIOR关键字的作用。还讨论了如何处理循环行, nocycle选项的使用,以及如何通过connect_by_iscycle和connect_by_isleaf判断循环行和叶子节点。此外,文章还展示了如何利用level伪列来表示层级,并通过实例展示了查询员工树结构的不同方法。

start with: 指定起始节点的条件

connect by: 指定父子行的条件关系

prior: 查询父行的限定符,格式: prior column1 = column2 or column1 = prior column2 and … ,

nocycle: 若数据表中存在循环行,那么不添加此关键字会报错,添加关键字后,便不会报错,但循环的两行只会显示其中的第一条

循环行: 该行只有一个子行,而且子行又是该行的祖先行

connect_by_iscycle: 前置条件:在使用了nocycle之后才能使用此关键字,用于表示是否是循环行,0表示否,1 表示是

connect_by_isleaf: 是否是叶子节点,0表示否,1 表示是

level: level伪列,表示层级,值越小层级越高,level=1为层级最高节点

-- 创建表
create table A_TEST_EMPLOYEE(
       emp_id number(18),
       lead_id number(18),
       emp_name varchar2(200),
       salary number(10,2),
       dept_no varchar2(8)
);

-- 添加数据
insert into A_TEST_EMPLOYEE values('1',0,'king','1000000.00','001');
insert into A_TEST_EMPLOYEE values('2',1,'jack','50500.00','002');
insert into A_TEST_EMPLOYEE values('3',1,'arise','60000.00','003');
insert into A_TEST_EMPLOYEE values('4',2,'scott','30000.00','002');
insert into A_TEST_EMPLOYEE values('5',2,'tiger','25000.00','002');
insert into A_TEST_EMPLOYEE values('6',3,'wudde','23000.00','003');
insert into A_TEST_EMPLOYEE values('7',3,'joker','21000.00','003');commit;
关系树状图

在这里插入图片描述

表结构和信息如下

--connect by 测试
select t.*, t.rowid from A_TEST_EMPLOYEE t;

在这里插入图片描述

查询以emp_id为1开始的节点的所有直属节点
select emp_id,lead_id,emp_name,prior emp_name as lead_name,salary
     from A_TEST_EMPLOYEE
     start with  emp_id=1
     connect by prior emp_id =  lead_id 
     

等价于

select emp_id,lead_id,emp_name,prior emp_name as lead_name,salary
     from A_TEST_EMPLOYEE
     start with  lead_id=0
     connect by prior emp_id =  lead_id 

查询结果如下
在这里插入图片描述

以emp_id为6的所有祖先节点
select emp_id,lead_id,emp_name,salary
       from A_TEST_EMPLOYEE
       start with emp_id = 6
       connect by prior lead_id = emp_id

查询结果如下
在这里插入图片描述

查询一个节点的叔叔伯父节点
with t as (
       select  A_TEST_EMPLOYEE.*,prior emp_name,level le
       from A_TEST_EMPLOYEE 
       start with lead_id = 0
       connect by lead_id=prior emp_id
       )
       select t.*
       from t 
       left join t tt on tt.emp_id=6
       where t.le = (tt.le-1)
       and t.emp_id not in (tt.lead_id)  

查询结果如下
在这里插入图片描述

查询族兄
with t as (
            select A_TEST_EMPLOYEE.*,prior emp_name,level le
           from A_TEST_EMPLOYEE 
           start with lead_id=0
           connect by lead_id= prior emp_id
       )
       select t.*
       from t  t
       left join t tt on tt.emp_id=6
       where t.le=tt.le and t.emp_id<>6;

查询结果如下
在这里插入图片描述

level伪列的使用,格式化层级
select lpad(' ',level*3,'   ')||emp_name as name, emp_id,lead_id,salary,level 
from a_test_employee 
start with lead_id = 0 
connect by prior emp_id = lead_id

查询结果如下
在这里插入图片描述

connect_by_root 查找根节点
select connect_by_root emp_id as root_emp_id,connect_by_root emp_name as root_emp_name, lpad(' ',level*3,'   ')||emp_name as name, emp_id,lead_id,salary,level
from a_test_employee 
where 1=1
and salary > 50000
start with lead_id = 0 
connect by prior emp_id = lead_id

查询结果如下
在这里插入图片描述

connect_by_isleaf 是否是叶子节点
select ate.*,level,connect_by_isleaf 
from a_test_employee ate
where 1=1
start with lead_id = 0
connect by prior emp_id = lead_id

查询结果如下
在这里插入图片描述

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值