CONNECT BY(ROWNUM|LEVEL) 特殊用法解释(试着解释)

本文详细介绍了Oracle中的CONNECT BY语句,用于递归查询,特别是ROWNUM和LEVEL的使用。通过实例展示了它们如何生成序列和进行复杂递归。通过层级查询,解释了递归的层次和限制条件,以及不同条件下数据的递归模式。此外,还讨论了CONNECT BY在特定条件下的控制,如level是否为null对递归的影响。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

with t as 
(select  rownum from dual t CONNECT BY rownum <4)
select t.*,rownum from t CONNECT BY rownum < 5;


CONNECT BY(ROWNUM|LEVEL) 特殊用法解释(试着解释)

CONNECT BY 是sql内部建立递归查询的方式,该方法在层级查询以及 ROWNUM|LEVEL是同理,都是递归到最后限定的值or ROWNUM|LEVEL。

1) CONNECT BY ROWNUM (简单递归),上述例子中 with子查询中使用 CONNECT BY(ROWNUM|LEVEL) 生成固定的序列数(并不复杂而且二者效果一样),
都是使用伪列生成数据(咱用结果为 1,2,3 ),第二层(外层)sql语句中如果是 CONNECT BY ROWNUM, 则语句递归方式为简单方式,
例中第二层CONNECT BY rownum <5, 即只递归第一个数字(或其他字符)使其伪列达到规定边界CONNECT BY rownum <5 , 后再列出剩余数字(剩余数字不受限定影响)。


with t as 
(select  level from dual t CONNECT BY level <4)

select t.*,level from t CONNECT BY level < 5;

2)CONNECT BY LEVEL(复杂递归), with子句与上述相同上述例子中 with子查询中使用 CONNECT BY(ROWNUM|LEVEL) 生成固定的序列数(并不复杂而且二者效果一样),
都是使用伪列生成数据(咱用结果为 1,2,3 )
第二层(外层)sql语句中如果是 CONNECT BY level, 则语句递归方式为复杂递归方式-即with语句结果中每个数都参与递归,并直至达到规定伪列level数
详细说明a:CONNECT BY level < 5 限定了sql中递归的层级数,如果以图形标示,一共有4层层级,第一层列出 1,2,3; 第二层 在第一层1的下面列出1,2,3,
在第一层2的下面列出1,2,3。。。。以此类推,直至第四层列出所有第三层中 1,2,3下的1,2,3。 完成递归


详细说明b:CONNECT BY level < 5 限定了sql中递归的层级数,如果以数据结果表标示, 
首先以第一个数1为根递归至第二层1,再至第三层1,最后到第四层(限定层)列出 1,2,3, 在返回第三层2,再下探到第四层1,2,3,循环反复,直至,每一层的1,2,3都被

递归到, 完成递归。

另一个例子

下面开始实验

复制代码

CREATE TABLE T(ID VARCHAR2(1 BYTE));

INSERT INTO T ( ID ) VALUES ( 'A'); 
INSERT INTO T ( ID ) VALUES ( 'B'); 
INSERT INTO T ( ID ) VALUES ( 'C'); 
COMMIT;

SQL> select id,level from t connect by level<2;
I LEVEL
- ----------
A 1
B 1
C 1
SQL> select id,level from t connect by level<3;
I LEVEL
- ----------
A 1
A 2
B 2
C 2
B 1
A 2
B 2
C 2
C 1
A 2
B 2
C 2
已选择12行。
SQL> select id,level from t connect by level<4;
I LEVEL
- ----------
A 1
A 2
A 3
B 3
C 3
B 2
A 3
B 3
C 3
C 2
A 3
B 3
C 3
B 1
A 2
A 3
B 3
C 3
B 2
A 3
B 3
C 3
C 2
A 3
B 3
C 3
C 1
A 2
A 3
B 3
C 3
B 2
A 3
B 3
C 3
C 2
A 3
B 3
C 3
已选择39行。

复制代码

我们很快可以找到其中的规律,假设表中有N条记录, 则记F(N,l)为select id,level from t connect by level<l 的结果集数目
那么,
F(N,1)=N
F(N,l) = F(N,l-1)*N+N

于是可以总结出
F(N,l)=∑power(N,p), p取值为[1,l)

要解释,也很容易:当连接条件不能限制记录之间的关系时每一条记录都可以作为自己或者其他记录的叶子
如下所示:
A          1
A          2
A          3
B          3
C          3
B          2
A          3
B          3
C          3
C          2
A          3
B          3
C          3

在这里,我们看到的是Oracle采用了深度优先的算法;

--------------update 2019/09/04 connect by 引申用法

当出现下述语句时,表明 level的限制条件 为判断是否有控制

默认情况下 level 只有1层, level is null 因此当level遍历一次之后,level 就是null;  就停止

反之   level is not null 因此当level遍历一次之后,level 无限增加;   数据也会无限循环;

with t as 
(select  level from dual t CONNECT BY level <4
union select null from dual )
select t.* , from t CONNECT BY level is null [not null];

refer to https://www.cnblogs.com/princessd8251/p/4064060.html;

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值