oracle+connect+by+level,oracle connect by用法篇
1、基本语法
select * from table [start with condition1] connect by [prior] id=parentid1
2
一般用来查找存在父子关系的数据,也就是树形结构的数据;其返还的数据也能够明确的区分出每一层的数据。
start with condition1 是用来限制第一层的数据,或者叫根节点数据;以这部分数据为基础来查找第二层数据,然后以第二层数据查找第三层数据以此类推。
connect by [prior] id=parentid 这部分是用来指明oracle在查找数据时以怎样的一种关系去查找;比如说查找第二层的数据时用第一层数据的id去跟表里面记录的parentid字段进行匹配,如果这个条件成立那么查找出来的数据就是第二层数据,同理查找第三层第四层…等等都是按这样去匹配。
prior还有一种用法:
select * from table [start with condition1] connect by id= [prior] parentid1
2
这种用法就表示从下往上查找数据,可以理解为从叶子节点往上查找父级几点,用第一层数据的parentid去跟表记录里面的id进行匹配,匹配成功那么查找出来的就是第二层数据;上面的那种就是从父级节点往下查找叶子节点。
其他特性
level关键字,代表树形结构中的层级编号;第一层是数字1,第二层数字2,依次递增。
CONNECT_BY_ROOT方法,能够获取第一层集结点结果集中的任意字段的值;例CONNECT_BY_ROOT(字段名)。
2、下面来贴两个例子
2.1 从根节点查找叶子节点
select t.*, level, CONNECT_BY_ROOT(id) from tab_test t start with t.id = 0 connect by prior t.id = t.fid;1
2
3
4
2.2 从叶子节点查找上层节点
--第一种,修改prior关键字位置 select t.*, level, CONNECT_BY_ROOT(id) from tab_test t start with t.id = 4 connect by t.id = prior t.fid; --第二种,prior关键字不动 调换后面的id=fid逻辑关系的顺序 select t.*, level, CONNECT_BY_ROOT(id) from tab_test t start with t.id = 4 connect by prior t.fid = t.id;1
2
3
4
5
6
7
8
9
10
11
3、写几个平常用到的其他一些用法
3.1 生成数字序列结果集
使用rownum实现1到10的序列。
select rownum from dual connect by rownum<=10;1
结果集如下:
使用level实现1到10的序列。
select level from dual connect by level<=10;1
结果集如下:
3.2 查询当前时间往前的12周的开始时间、结束时间、第多少周
select sysdate - (to_number(to_char(sysdate - 1, 'd')) - 1) -
(rownum - 1) * 7 as startDate,
sysdate + (7 - to_number(to_char(sysdate - 1, 'd'))) -
(rownum - 1) * 7 as endDate,
to_number(to_char(sysdate, 'iw')) - rownum + 1 as weekIndex from dual connect by level<= 12;--将level改成rownum可以实现同样的效果1
2
3
4
5
6
7
d 表示一星期中的第几天
iw 表示一年中的第几周
3.3 字符串分割,由一行变为多行
比如说分割01#02#03#04这种有规律的字符串
select REGEXP_SUBSTR('01#02#03#04', '[^#]+', 1, rownum) as newport from dual connect by rownum <= REGEXP_COUNT('01#02#03#04', '[^#]+');1
2
4、省略prior关键字时数据的返回策略
构造一个结果集,其中包含两条数据;然后查询level为1,2,3层的数据。
select t.*, level from (select 1 as num from dual union select 2 as num from dual
) t connect by level <= 3;1
2
3
4
5
6
从上面截图的结果可以看出来省略prior关键字时第1层的数据就是初始结果集,第2层的数据是初始结果集的两倍,第3层的数据是初始结果集的3倍;假设初始结果集的记录为n条,查询m层的记录,则返回的记录数就是:
条记录。
在省略prior关键字对数据进行操作时需要特别注意,返回的数据不一定是你所期望的那样。
5、下面再看看几个例子,针对多条结果集当省略prior关键字时怎样获得正确的返回结果
5.1 有下面一个结果集
想要实现1-5,20-30的数据递增返回1、2、3、4、5、20、21、22、23、24、25、26、27、28、29、30总共16条记录。
SQL如下:
with temp0 as ( select t.range_num,
REGEXP_SUBSTR(t.range_num, '[^-]+', 1, 1) minNum, --最小num
REGEXP_SUBSTR(t.range_num, '[^-]+', 1, 2) maxNum --最大num from range_table t
) select t1.range_num ,t2.lv from temp0 t1 join ( select level lv from dual CONNECT BY LEVEL <= (select max(maxNum) from temp0 )
) t2 on (t2.lv >=t1.minNum and t2.lv <=t1.maxNum);1
2
3
4
5
6
7
8
9
10
11
12
13
上面的sql中是先求出range_num的最大值与最小值,然后利用connect by 特性生成数值序列结果集,最后让两个结果集关联得到需要的结果。
5.2 再看稍微复杂的结果集,输出结果格式跟上面一样
SQL如下:
with temp0 as ( select b.range_num,
REGEXP_SUBSTR(b.range_num, '[^,]+', 1, c.lv) as newport,
REGEXP_SUBSTR(REGEXP_SUBSTR(b.range_num, '[^,]+', 1, c.lv), '[^-]+', 1, 1) as minNum,
REGEXP_SUBSTR(REGEXP_SUBSTR(b.range_num, '[^,]+', 1, c.lv), '[^-]+', 1, 2) as maxNum from (select regexp_count(a.range_num, '[^,]+') AS cnt,
range_num from range_table a) b join (select LEVEL lv from dual CONNECT BY LEVEL <= 50) c
--这里的50表示的是range_num通过,分割后的数量,这里写死了50也可以sql动态max出来 on c.lv <= b.cnt
) select t1.range_num,t2.lv from temp0 t1 join ( select level lv from dual CONNECT BY LEVEL <= ( select max(to_number(maxNum)) from temp0
)
) t2 on ((t2.lv >=t1.minNum and t2.lv <=t1.maxNum));1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
oracle+connect+by+level,oracle connect by用法篇相关推荐
- oracle级联查询 level,ORACLE 数据库的级联查询 一句sql搞定(部门多级)
select t.destination from airline t start with origin='CAN' connect by prior destination = origin; 查 ...
- [Oracle SQL]最基本的connect by的用法
最基本的connect by的用法: 需求1: 我需要下钻所有的树得到level和parent_name create table test_lvl1 (id number, parent_id nu ...
- oracle connect by用法篇
原文链接:https://blog.csdn.net/wang_yunj/article/details/51040029 1.基本语法 select * from table [start with ...
- oracle中start with和connect by的用法理解
Oracle中start with和connect by 用法理解 转自:http://www.blogjava.net/xzclog/archive/2010/03/05/314642.html,多 ...
- oracle——connect by level
connect by 层级查询 用于存在父子,祖孙,上下级等层级关系的数据表进行层级查询 语法格式: { CONNECT BY [ NOCYCLE ] condition [AND condition ...
- mysql6支持connect by_mysql 实现oracle start with connect by递归
在Oracle 中我们知道有一个 Hierarchical Queries 通过CONNECT BY 我们可以方便的查了所有当前节点下的所有子节点.但很遗憾,在MySQL的目前版本中还没有对应的功能. ...
- Oracle中start with xx connect by prior 语句解析
Oracle中start with xx connect by prior 语句解析 Oracle这种的start with语句主要对B型树的数据进行递归查询.可以指定数据树上的任一节点,然后查找 ...
- ORACLE报错:ORA-01788: CONNECT BY clause required in this query block
ORACLE报错:ORA-01788: CONNECT BY clause required in this query block 原因:oracle表中有字段level(关键字),直接查询该字段报 ...
- oracle 递归查询,Oracle递归查询(start with…connect by prior)
查询基本结构: select - from table_name start with 条件1 connect by 条件2 1.建测试用表 1 create table test_prior( 2 ...
最新文章
- Java并发编程之CountDownLatch、CyclicBarrier和Semaphore
- 特征级融合_遥感影像融合的方法有哪些
- VB讲课笔记01:VB6.0安装与启动
- 在weblogic上配置数据源
- cmder的下载安装
- 一个js内存泄露的好例子
- umijs 出错 TypeError: rawData.some is not a function
- p5.js之Q版人物绘制
- Error starting Tomcat context. Exception
- Python自动化修改word实例
- 阿里云AI解决方案-身份证图像识别
- android开发者mac(含M1芯片)电脑全新配置2022
- 华硕电脑装linux黑屏,华硕电脑更新显卡后开机黑屏应该怎么解决
- C++代码破解LOL人物基址特征,基址在源码里面自己找!
- Android基础——Alarm
- idea 项目能运行,但是代码冒红-解决措施
- python——简单通讯录
- 理解LSTMs (Long Short Term Memory Networks)
- NetworkX画图:nx.draw_networkx(函数详解)
- 指令在Vue.js 2.0中的变化
热门文章
- 无法将类型为“System.__ComObject”的 COM 对象强制转换为类类型“mshtml.HTMLInputElementClass...
- 完整的POM文档内容
- EasyUI Tree判断节点是否是叶
- 以下是关于ASP.NET中保存各种信息的对象的比较,理解这些对象的原理,对制作完善的程序来说是相当有必要的(摘至互联网,并非原创--xukunping)...
- jQuery同步Ajax带来的UI线程阻塞问题及解决办法
- Android开发一 什么是3G
- 怎么在搭建Android开发环境?
- C# 如何生成一个时间戳
- Java中JSON字符串与java对象的互换实例详解
- php 7连接mysql数据库