Oracle的学习心得和知识总结(五)|Oracle数据库 Connect By 技术详解

本文涉及的产品
云原生数据库 PolarDB MySQL 版,通用型 2核4GB 50GB
云原生数据库 PolarDB PostgreSQL 版,标准版 2核4GB 50GB
简介: Oracle的学习心得和知识总结(五)|Oracle数据库 Connect By 技术详解

目录结构

注:提前言明 本文借鉴了以下博主、书籍或网站的内容,其列表如下:


1、Oracle Database Documentation 官方文档,点击前往

2、oracle connect by用法篇,点击前往

3、参考书籍:《涂抹Oracle 三思笔记之一步一步学Oracle》

4、参考书籍:《Oracle Database 11g数据库管理艺术》

5、ORACLE分层查询start with&connect by,点击前往

6、oracle 进阶 connect by 和level 的用法,点击前往

7、参考书籍:《Oracle Database 9i10g11g编程艺术深入数据库体系结构(第2版)》

8、树状数据库表:Oracle中start with…connect by prior子句用法,点击前往

9、Hierarchical Queries,点击前往


Oracle数据库 Connect By 技术

本人博客严正声明

文章快速说明索引

Connect By的简介

Connect By的使用

Connect By的总结


image.png

image.png

本人博客严正声明

是这样的,熟悉在下的小伙伴们都知道 我写博客主要目的就是分享和学习总结。至于CSDN的排名 排名什么的,我并不是很在意!


一来 很不喜欢标题党

二来 更反感灌些水文

三来 痛恨无下限抄袭

本人博客都是认认真真写的,结果在CSDN并没有什么的太大的名气 关注度什么的也不高!前些天 一位好心的粉丝私聊了在下,反而一名某平台的哥们儿 快把我的《PostgreSQL的学习心得和知识总结》都给照搬过去了,甚至一个字都不改(连同在下 都是只字不提 好歹稍微提一下呀)!!!


实在是太过分,后来经过(友好)协商,现已经全部删除了!


本人是做PostgreSQL内核开发的,深感当下学风不正 大家都很浮躁,一向踏踏实实深耕的并不是很多!因为写代码这件事情上,欺骗不了任何人!本本分分老老实实地写好代码做好学问十分不易,容不得掺沙子和造假!这里把我喜欢的一句话送给各位以共勉:


非淡泊无以明志,

非宁静无以致远!


文章快速说明索引

学习目标:


目的:因为接下来想在PostgreSQL数据库内核上实践实现一下 Connect By操作 ,但是就目前而言 PostgreSQL尚不支持此功能。PostgreSQL社区的爱好者和一些基于PostgreSQL的数据库公司也曾分享过一些 类似的 Connect By的简易实现,大家有兴趣也可以去看看!本文主要记录在Oracle数据库上面的 Connect By 功能的使用体验和基于开发者的设计思考,以期在PostgreSQL数据库上面支持此功能!这里主要是学习以及介绍Oracle数据库 Connect By 功能的注意事项等,基于PostgreSQL数据库的功能开发等日后开发完成之后 由新博客进行介绍和分享!


学习内容:(详见目录)


1、Oracle数据库的Connect By技术


学习时间:


2021年06月12日 16:36:04


学习产出:


1、Oracle数据库 Connect By 技术学习

2、CSDN 技术博客 1篇

3、PostgreSQL数据库 Connect By 功能实现设计思考


注:下面我们所有的学习环境是Centos7+PostgreSQL13.1+Oracle11g+MySQL5.7

postgres=# select version();
                                   version                                   
-----------------------------------------------------------------------------
 PostgreSQL 13.1 on x86_64-pc-linux-gnu, compiled by gcc (GCC) 7.1.0, 64-bit
(1 row)
postgres=#
#-----------------------------------------------------------------------------#
SQL> select * from v$version;
BANNER
--------------------------------------------------------------------------------
Oracle Database 11g Enterprise Edition Release 11.2.0.1.0 - 64bit Production
PL/SQL Release 11.2.0.1.0 - Production
CORE  11.2.0.1.0  Production
TNS for Linux: Version 11.2.0.1.0 - Production
NLSRTL Version 11.2.0.1.0 - Production
SQL>
#-----------------------------------------------------------------------------#
mysql> select version();
+-----------+
| version() |
+-----------+
| 5.7.19    |
+-----------+
1 row in set (0.06 sec)
mysql>

Connect By的简介

在Oracle数据库中,如果表包含分层数据,则可以使用Hierarchy query子句按分层顺序选择行。其语法格式如下:

{ CONNECT BY [ NOCYCLE ] condition [ START WITH condition ]
| START WITH condition CONNECT BY [ NOCYCLE ] condition
}

image.png

解释一下上图的逻辑(来自于Oracle的官方文档):


以上两种SQL都是可以的

condition:条件 指定一个或多个表达式和逻辑(布尔)运算符的组合,并返回TRUE、FALSE或UNKNOWN值

start with:指定起始节点的条件 或 指定层次结构的根行

connect by:指定层次结构的父行和子行之间的条件关系

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

nocycle: 若数据表中存在循环行,那么不添加此关键字会报错,添加关键字后,便不会报错,但循环的两行只会显示其中的第一条。(NOCYCLE参数指示Oracle数据库从查询返回行,即使数据中存在CONNECT BY循环。将此参数与CONNECT_BY_ISCYCLE伪列一起使用,以查看哪些行包含循环。有关更多信息,请参阅 CONNECT_BY_ISCYCLE 伪列。)

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

在分层查询中,条件中的一个表达式必须使用 PRIOR 运算符限定才能引用父行:

... PRIOR expr = expr
or
... expr = PRIOR expr

如果CONNECT BY condition条件是复合条件,则只有一个条件需要 PRIOR 运算符,尽管可以有多个 PRIOR 条件:

CONNECT BY last_name != 'King' AND PRIOR employee_id = manager_id ...
CONNECT BY PRIOR employee_id = manager_id and PRIOR account_mgr_id = customer_id ...

Connect By 是结构化查询中用到的,更亲切的语法如下:

select targetlist 
  from tablename [where condition0]
  start with condition1
  connect by condition2

详细解释一下:


将一个树状结构存储在一张表里,比如一个表中存在两个字段:keyid,parent_keyid那么通过表示每一条记录的parent是谁,就可以形成一个树状结构


用上述语法的查询可以取得这棵树的所有记录


1、condition1 是根结点的限定语句,当然可以放宽限定条件,以取得多个根结点,实际就是多棵树

2、condition2 是连接条件,其中用PRIOR表示上一条记录,比如 CONNECT BY PRIOR keyid = parent_keyid就是说上一条记录的keyid 是本条记录的parent_keyid,即本记录的父亲是上一条记录


接下来,我们这里准备一下备用的数据如下:

-- 创建表
create table player(keyid int,parent_keyid int,name varchar(16),salary int,sex varchar(4));
-- 添加数据
insert into player values(1,0,'zhangsan','1000000','f');
insert into player values(2,1,'lisi','50500','m');
insert into player values(3,1,'wangwu','60000','m');
insert into player values(4,1,'houzi','65000','m');
insert into player values(5,2,'maliu','30000','f');
insert into player values(6,2,'liuqi','25000','m');
insert into player values(7,4,'gouba','23000','m');
insert into player values(8,4,'dujiu','21000','f');

查询&树状展示如下:

SQL> select t.*, t.rowid from player t;
     KEYID PARENT_KEYID NAME           SALARY SEX  ROWID
---------- ------------ ---------- ---------- ---- ------------------
         1            0 zhangsan      1000000 f    AAATSIAABAAAVRhAAA
         2            1 lisi            50500 m    AAATSIAABAAAVRhAAB
         3            1 wangwu          60000 m    AAATSIAABAAAVRhAAC
         4            1 houzi           65000 m    AAATSIAABAAAVRhAAD
         5            2 maliu           30000 f    AAATSIAABAAAVRhAAE
         6            2 liuqi           25000 m    AAATSIAABAAAVRhAAF
         7            4 gouba           23000 m    AAATSIAABAAAVRhAAG
         8            4 dujiu           21000 f    AAATSIAABAAAVRhAAH
8 rows selected.
SQL>

image.png

Connect By的使用

下面来看一下,在Oracle数据库中的一些常见使用:

第一个:查询以keyid为1开始的节点的所有子孙节点

SQL> select keyid,parent_keyid,name,prior name,salary
  2       from player
  3       start with keyid=1
  4       connect by prior keyid = parent_keyid;
     KEYID PARENT_KEYID NAME       PRIORNAME            SALARY
---------- ------------ ---------- ---------------- ----------
         1            0 zhangsan                       1000000
         2            1 lisi       zhangsan              50500
         5            2 maliu      lisi                  30000
         6            2 liuqi      lisi                  25000
         3            1 wangwu     zhangsan              60000
         4            1 houzi      zhangsan              65000
         7            4 gouba      houzi                 23000
         8            4 dujiu      houzi                 21000
8 rows selected.
# 该示例添加了一个 START WITH 子句来为层次结构指定一个根行,并添加一个使用 SIBLINGS 关键字的 ORDER BY 子句来保留层次结构内的排序:
SQL> select keyid,parent_keyid,name,prior name,salary,LEVEL le
  2         from player
  3         start with keyid=1
  4         connect by prior keyid = parent_keyid
  5         ORDER SIBLINGS BY name;
     KEYID PARENT_KEYID NAME       PRIORNAME            SALARY         LE
---------- ------------ ---------- ---------------- ---------- ----------
         1            0 zhangsan                       1000000          1
         4            1 houzi      zhangsan              65000          2
         8            4 dujiu      houzi                 21000          3
         7            4 gouba      houzi                 23000          3
         2            1 lisi       zhangsan              50500          2
         6            2 liuqi      lisi                  25000          3
         5            2 maliu      lisi                  30000          3
         3            1 wangwu     zhangsan              60000          2
8 rows selected.
SQL>

实质上,这个等价于:

SQL> select keyid,parent_keyid,name,prior name,salary
  2       from player
  3       start with  parent_keyid=0
  4       connect by prior keyid =  parent_keyid;
     KEYID PARENT_KEYID NAME       PRIORNAME            SALARY
---------- ------------ ---------- ---------------- ----------
         1            0 zhangsan                       1000000
         2            1 lisi       zhangsan              50500
         5            2 maliu      lisi                  30000
         6            2 liuqi      lisi                  25000
         3            1 wangwu     zhangsan              60000
         4            1 houzi      zhangsan              65000
         7            4 gouba      houzi                 23000
         8            4 dujiu      houzi                 21000
8 rows selected.
SQL>

注意:在分层查询中,不要指定 ORDER BY 或 GROUP BY,因为它们会覆盖 CONNECT BY 结果的分层顺序。 如果要对同一父级的同级行进行排序,请使用 ORDER SIBLINGS BY 子句。 请参阅 order_by_clause(如上所示)。


第二个:查询以keyid为2开始的节点的所有子孙节点

SQL> select keyid,parent_keyid,name,prior name,salary
  2       from player
  3       start with keyid=2
  4       connect by prior keyid = parent_keyid;
     KEYID PARENT_KEYID NAME       PRIORNAME            SALARY
---------- ------------ ---------- ---------------- ----------
         2            1 lisi                             50500
         5            2 maliu      lisi                  30000
         6            2 liuqi      lisi                  25000
SQL>

第三个:查询以keyid为3开始的节点的所有子孙节点

SQL> select keyid,parent_keyid,name,prior name,salary
  2       from player
  3       start with keyid=3
  4       connect by prior keyid = parent_keyid;
     KEYID PARENT_KEYID NAME       PRIORNAME            SALARY
---------- ------------ ---------- ---------------- ----------
         3            1 wangwu                           60000
SQL>

从上面三个例子可得:


在扫描树结构表时,需要依此访问树结构的每个节点,一个节点只能访问一次,其访问的步骤如下:

第一步:从根节点(广义上的根)开始

第二步:访问该节点

第三步:判断该节点有无未被访问的子节点,若有,则转向它最左侧的未被访问的子节,并执行第二步,否则执行第四步

第四步:若该节点为根节点,则访问完毕,否则执行第五步

第五步:返回到该节点的父节点,并执行第三步骤

我们大家细细地品一下,上述扫描整个树结构的过程就是一个先序遍历树的过程。这一部分内容可以参见本人博客:二叉树的四种遍历方式(递归与非递归实现),点击前往


完整概括一下(Oracle官方文档):


Oracle对分层查询的处理如下:


如果存在join,则首先计算join,无论该join是在FROM子句中指定的还是使用WHERE子句谓词

计算CONNECT BY条件

计算其余的WHERE子句谓词

然后Oracle使用这些评估的信息来形成层次结构,步骤如下:


Oracle选择层次结构的根行—那些满足START WITH条件的行

Oracle选择每个根行的子行。每个子行必须满足一个根行的CONNECT BY条件

Oracle选择连续的子行。Oracle首先选择步骤2中返回的行的子行,然后再选择这些子行的子行,以此类推。Oracle总是通过对当前父行计算CONNECT by条件来选择子行

如果查询包含一个没有连接的WHERE子句,那么Oracle将从层次结构中删除所有不满足WHERE子句条件的行。Oracle对每一行单独计算这个条件,而不是删除不满足条件的行的所有子行

Oracle按照下图所示的顺序返回行。在图表中,孩子出现在父母的下方


image.png

image.png

为了找到父行的子行,Oracle计算父行CONNECT BY条件的PRIOR表达式和表中每一行的另一个表达式。其中条件为真的行是父行的子行。当然CONNECT BY条件可以包含其他条件,以进一步筛选查询所选择的行

如果CONNECT BY条件导致层次结构中出现循环,那么Oracle将返回一个错误。

循环:如果一行同时是另一行的父行(或祖父行或直系祖先行)和子行(或孙子行或直系后代行),则会发生循环(这个后面详细案例介绍)。


第四个:查询以keyid为6的所有祖先节点

SQL> select keyid,parent_keyid,name,salary
  2         from player
  3         start with keyid = 6
  4         connect by prior parent_keyid = keyid;
     KEYID PARENT_KEYID NAME           SALARY
---------- ------------ ---------- ----------
         6            2 liuqi           25000
         2            1 lisi            50500
         1            0 zhangsan      1000000
SQL>

这里的这个就是上面三个的逆过程:

  1. 找到 keyid = 6 这一行,得到其 parent_keyid=2
  2. 由 connect by 可得,(从其他行中)继续找 keyid是2的行,找到了 重复1和2

第五个:查询以keyid为6节点的叔叔伯父节点

SQL> select player.*,prior name,level le
  2          from player 
  3          start with parent_keyid = 0
  4          connect by parent_keyid=prior keyid;
     KEYID PARENT_KEYID NAME           SALARY SEX  PRIORNAME                LE
---------- ------------ ---------- ---------- ---- ---------------- ----------
         1            0 zhangsan      1000000 f                              1
         2            1 lisi            50500 m    zhangsan                  2
         5            2 maliu           30000 f    lisi                      3
         6            2 liuqi           25000 m    lisi                      3
         3            1 wangwu          60000 m    zhangsan                  2
         4            1 houzi           65000 m    zhangsan                  2
         7            4 gouba           23000 m    houzi                     3
         8            4 dujiu           21000 f    houzi                     3
8 rows selected.
SQL>
SQL> with t as (
  2         select  player.*,prior name,level le
  3         from player 
  4         start with parent_keyid = 0
  5         connect by parent_keyid=prior keyid)
  6         select t.*            
  7         from t
  8         left join t tt on tt.keyid=6
  9         where t.le = (tt.le-1) #找到上一层
 10         and t.keyid not in (tt.parent_keyid);
     KEYID PARENT_KEYID NAME           SALARY SEX  PRIORNAME                LE
---------- ------------ ---------- ---------- ---- ---------------- ----------
         4            1 houzi           65000 m    zhangsan                  2
         3            1 wangwu          60000 m    zhangsan                  2
SQL>

第六个:查询以keyid为6节点的族兄节点

SQL> with t as (
  2             select player.*,prior name,level le
  3             from player 
  4             start with parent_keyid=0
  5             connect by parent_keyid= prior keyid
  6         )
  7         select t.*
  8         from t  t
  9         left join t tt on tt.keyid=6
 10         where t.le=tt.le and t.keyid<>6; # 同一层
     KEYID PARENT_KEYID NAME           SALARY SEX  PRIORNAME                LE
---------- ------------ ---------- ---------- ---- ---------------- ----------
         8            4 dujiu           21000 f    houzi                     3
         7            4 gouba           23000 m    houzi                     3
         5            2 maliu           30000 f    lisi                      3
SQL>

第七个:level伪列的使用,格式化层级

SQL> select player.*,prior name,level le
  2          from player 
  3          start with parent_keyid = 0
  4          connect by parent_keyid=prior keyid;
     KEYID PARENT_KEYID NAME           SALARY SEX  PRIORNAME                LE
---------- ------------ ---------- ---------- ---- ---------------- ----------
         1            0 zhangsan      1000000 f                              1
         2            1 lisi            50500 m    zhangsan                  2
         5            2 maliu           30000 f    lisi                      3
         6            2 liuqi           25000 m    lisi                      3
         3            1 wangwu          60000 m    zhangsan                  2
         4            1 houzi           65000 m    zhangsan                  2
         7            4 gouba           23000 m    houzi                     3
         8            4 dujiu           21000 f    houzi                     3
8 rows selected.
SQL>
SQL> column name   format a30
SQL> select lpad(' ',level*3,' ')||name as name,keyid,parent_keyid,level 
  2     from player
  3     start with parent_keyid = 0 
  4     connect by prior keyid = parent_keyid;
NAME                                KEYID PARENT_KEYID      LEVEL
------------------------------ ---------- ------------ ----------
   zhangsan                             1            0          1
      lisi                              2            1          2
         maliu                          5            2          3
         liuqi                          6            2          3
      wangwu                            3            1          2
      houzi                             4            1          2
         gouba                          7            4          3
         dujiu                          8            4          3
8 rows selected.
SQL>

第八个:CONNECT_BY_ROOT的使用

CONNECT_BY_ROOT 操作的功能就是获取根节点记录的字段信息

SQL> select connect_by_root keyid as root_emp_id,connect_by_root name as root_emp_name, lpad(' ',level*3,' ')||name as name, keyid,parent_keyid,salary,level
  2     from player 
  3     start with parent_keyid = 0
  4     connect by prior keyid = parent_keyid;
ROOT_EMP_ID ROOT_EMP_NAME    NAME                                KEYID PARENT_KEYID     SALARY      LEVEL
----------- ---------------- ------------------------------ ---------- ------------ ---------- ----------
          1 zhangsan            zhangsan                             1            0    1000000          1
          1 zhangsan               lisi                              2            1      50500          2
          1 zhangsan                  maliu                          5            2      30000          3
          1 zhangsan                  liuqi                          6            2      25000          3
          1 zhangsan               wangwu                            3            1      60000          2
          1 zhangsan               houzi                             4            1      65000          2
          1 zhangsan                  gouba                          7            4      23000          3
          1 zhangsan                  dujiu                          8            4      21000          3
8 rows selected.
SQL> select connect_by_root keyid as root_emp_id,connect_by_root name as root_emp_name, lpad(' ',level*3,' ')||name as name, keyid,parent_keyid,salary,level
  2      from player where  1=1 and salary > 50000
  3      start with parent_keyid = 0
  4     connect by prior keyid = parent_keyid;
ROOT_EMP_ID ROOT_EMP_NAME    NAME                                KEYID PARENT_KEYID     SALARY      LEVEL
----------- ---------------- ------------------------------ ---------- ------------ ---------- ----------
          1 zhangsan            zhangsan                             1            0    1000000          1
          1 zhangsan               lisi                              2            1      50500          2
          1 zhangsan               wangwu                            3            1      60000          2
          1 zhangsan               houzi                             4            1      65000          2
SQL>

这里说的根仍旧是广义上的root,示例如下:

SQL> select connect_by_root keyid as root_emp_id,connect_by_root name as root_emp_name, lpad(' ',level*3,' ')||name as name, keyid,parent_keyid,salary,level
  2     from player
  3      start with keyid = 2
  4      connect by prior keyid = parent_keyid;
ROOT_EMP_ID ROOT_EMP_NAME    NAME                                KEYID PARENT_KEYID     SALARY      LEVEL
----------- ---------------- ------------------------------ ---------- ------------ ---------- ----------
          2 lisi                lisi                                 2            1      50500          1
          2 lisi                   maliu                             5            2      30000          2
          2 lisi                   liuqi                             6            2      25000          2
SQL>

CONNECT BY条件和PRIOR表达式都可以采用不相关子查询的形式。但是CURRVAL和NEXTVAL不是有效的PRIOR表达式,因此PRIOR表达式不能引用序列。


通过使用CONNECT_BY_ROOT操作符来限定选择列表中的列,可以进一步细化层次查询。这个操作符扩展了层次结构查询的CONNECT BY [PRIOR]条件的功能,(通过)不仅返回层次结构中的直接父行,而且返回层次结构中的所有祖先行(的方式)。

SQL> column name   format a20
SQL> select connect_by_root keyid as root_emp_id,connect_by_root name as root_emp_name, lpad(' ',level*3,' ')||name as name, keyid,parent_keyid,salary,level
  2      from player
  3      start with keyid = 6
  4       connect by prior parent_keyid = keyid;
ROOT_EMP_ID ROOT_EMP_NAME    NAME                      KEYID PARENT_KEYID     SALARY      LEVEL
----------- ---------------- -------------------- ---------- ------------ ---------- ----------
          6 liuqi               liuqi                      6            2      25000          1
          6 liuqi                  lisi                    2            1      50500          2
          6 liuqi                     zhangsan             1            0    1000000          3
SQL>

第九个:CONNECT_BY_ISLEAF 是否是叶子节点

通过这个伪列,可以判断当前的记录是否是树的叶节点

SQL> select p.*,level,CONNECT_BY_ISLEAF
  2     from player p
  3     start with parent_keyid = 0
  4     connect by prior keyid = parent_keyid;
     KEYID PARENT_KEYID NAME                               SALARY SEX       LEVEL CONNECT_BY_ISLEAF
---------- ------------ ------------------------------ ---------- ---- ---------- -----------------
         1            0 zhangsan                          1000000 f             1                 0
         2            1 lisi                                50500 m             2                 0
         5            2 maliu                               30000 f             3                 1
         6            2 liuqi                               25000 m             3                 1
         3            1 wangwu                              60000 m             2                 1
         4            1 houzi                               65000 m             2                 0
         7            4 gouba                               23000 m             3                 1
         8            4 dujiu                               21000 f             3                 1
8 rows selected.
SQL>

注:CONNECT_BY_ISCYCLE 前置条件:在使用了nocycle之后才能使用此关键字,用于表示是否是循环行,0表示否,1 表示是。(这里关于 树形成环 的问题下面详谈)


Connect By的总结

树结构的描述

树形结构的数据存放在表中,数据之间的层次关系即父子关系,通过表中的列与列间的关系来描述,如player表中的keyid,parent_keyid。在表的每一行中都有一个表示父节点的parent_keyid(除根节点外),指向每个节点的父节点,就可以确定整个树结构。


SELECT语句中的CONNECT BY子句说明每行数据将是按层次顺序检索,并规定将表中的数据连入树型结构的关系中。PRIORY运算符必须放置在连接关系的两列中某一个的前面。对于节点间的父子关系,PRIOR运算符在的一侧表示父节点,另一侧表示子节点,从而确定查找树结构是的顺序是自顶向下还是自底向上。在连接关系中,除了可以使用列名外,还允许使用列表达式。START WITH 子句为可选项,用来标识哪个节点作为查找树型结构的根节点。若该子句被省略,则表示所有满足查询条件的行作为根节点。


START WITH: 不但可以指定一个根节点,还可以指定多个根节点。


关于PRIOR

运算符PRIOR被放置于等号前后的位置,决定着查询时的检索顺序。


当PRIOR被置于CONNECT BY子句中等号的前面时,则强制从根节点到叶节点的顺序检索,即由父节点向子节点方向通过树结构,我们称之为自顶向下的方式。如:connect by prior keyid = parent_keyid

当PIROR运算符被置于CONNECT BY 子句中等号的后面时,则强制从叶节点到根节点的顺序检索,即由子节点向父节点方向通过树结构,我们称之为自底向上的方式。例如:connect by prior parent_keyid = keyid

在这种方式中也应指定一个开始的节点。


PRIOR是一元运算符,与一元算术运算符+和-具有相同的优先级;它计算分层查询中当前行的父行的紧接表达式。


PRIOR在用相等操作符比较列值时最常用。(PRIOR关键字可以在操作符的任意一侧)


PRIOR会导致Oracle使用列中父行的值。在CONNECT BY子句中,除了等号(=)以外的操作符理论上是可能的。然而,由这些其他操作符创建的条件可能导致通过可能的组合形成一个无限循环。在这种情况下,Oracle在运行时检测到循环并返回一个错误。


定义查找起始节点

在自顶向下查询树结构时,不但可以从根节点开始,还可以定义任何节点为起始节点,以此开始向下查找。这样查找的结果就是以该节点为开始的结构树的一枝。


使用LEVEL

在具有树结构的表中,每一行数据都是树结构中的一个节点,由于节点所处的层次位置不同,所以每行记录都可以有一个层号。层号根据节点与根节点的距离确定。不论从哪个节点开始,该起始根节点的层号始终为1,根节点的子节点为2, 依此类推。


节点和分支的裁剪

在对树结构进行查询时,可以去掉表中的某些行,也可以剪掉树中的一个分支,使用WHERE子句来限定树型结构中的单个节点,以去掉树中的单个节点,但它却不影响其后代节点(自顶向下检索时)或前辈节点(自底向顶检索时)。


排序显示

像在其它查询中一样,在树结构查询中也可以使用ORDER BY 子句,改变查询结果的显示顺序,而不必按照遍历树结构的顺序。


树形成环

我们这里先构造一个内含循环的树形结果来探讨一下该问题,如下:


image.png

image.png

SQL> select * from player;
     KEYID PARENT_KEYID NAME           SALARY SEX
---------- ------------ ---------- ---------- ----
         1            0 zhangsan      1000000 f
         2            1 lisi            50500 m
         3            1 wangwu          60000 m
         4            1 houzi           65000 m
         5            2 maliu           30000 f
         6            2 liuqi           25000 m
         7            4 gouba           23000 m
         8            4 dujiu           21000 f
         4            8 xunhuan         22222 m
9 rows selected.
SQL>

示例如下:

SQL> select keyid,parent_keyid,name,prior name,salary,LEVEL le
  2         from player
  3         start with keyid=1
  4         connect by prior keyid = parent_keyid;
ERROR:
ORA-01436: CONNECT BY loop in user data
no rows selected
SQL>

在这样的情况下,需要使用NOCYCLE 参数, CONNECT BY 条件中的 NOCYCLE 参数使 Oracle 不顾循环而返回行。CONNECT_BY_ISCYCLE 伪列显示哪些行包含循环:

SQL> select keyid,parent_keyid,name,prior name,salary,LEVEL le,CONNECT_BY_ISCYCLE "Cycle"
  2         from player
  3         start with keyid=1
  4         connect by NOCYCLE prior keyid = parent_keyid;
     KEYID PARENT_KEYID NAME       PRIORNAME            SALARY         LE      Cycle
---------- ------------ ---------- ---------------- ---------- ---------- ----------
         1            0 zhangsan                       1000000          1          0
         2            1 lisi       zhangsan              50500          2          0
         5            2 maliu      lisi                  30000          3          0
         6            2 liuqi      lisi                  25000          3          0
         3            1 wangwu     zhangsan              60000          2          0
         4            1 houzi      zhangsan              65000          2          0
         7            4 gouba      houzi                 23000          3          0
         8            4 dujiu      houzi                 21000          3          1
8 rows selected.
SQL>


相关实践学习
使用PolarDB和ECS搭建门户网站
本场景主要介绍基于PolarDB和ECS实现搭建门户网站。
阿里云数据库产品家族及特性
阿里云智能数据库产品团队一直致力于不断健全产品体系,提升产品性能,打磨产品功能,从而帮助客户实现更加极致的弹性能力、具备更强的扩展能力、并利用云设施进一步降低企业成本。以云原生+分布式为核心技术抓手,打造以自研的在线事务型(OLTP)数据库Polar DB和在线分析型(OLAP)数据库Analytic DB为代表的新一代企业级云原生数据库产品体系, 结合NoSQL数据库、数据库生态工具、云原生智能化数据库管控平台,为阿里巴巴经济体以及各个行业的企业客户和开发者提供从公共云到混合云再到私有云的完整解决方案,提供基于云基础设施进行数据从处理、到存储、再到计算与分析的一体化解决方案。本节课带你了解阿里云数据库产品家族及特性。
相关文章
|
27天前
|
存储 Oracle 关系型数据库
数据库数据恢复—ORACLE常见故障的数据恢复方案
Oracle数据库常见故障表现: 1、ORACLE数据库无法启动或无法正常工作。 2、ORACLE ASM存储破坏。 3、ORACLE数据文件丢失。 4、ORACLE数据文件部分损坏。 5、ORACLE DUMP文件损坏。
86 11
|
2月前
|
Oracle 关系型数据库 数据库
Oracle数据恢复—Oracle数据库文件有坏快损坏的数据恢复案例
一台Oracle数据库打开报错,报错信息: “system01.dbf需要更多的恢复来保持一致性,数据库无法打开”。管理员联系我们数据恢复中心寻求帮助,并提供了Oracle_Home目录的所有文件。用户方要求恢复zxfg用户下的数据。 由于数据库没有备份,无法通过备份去恢复数据库。
|
2月前
|
存储 Oracle 关系型数据库
oracle数据恢复—Oracle数据库文件大小变为0kb的数据恢复案例
存储掉盘超过上限,lun无法识别。管理员重组存储的位图信息并导出lun,发现linux操作系统上部署的oracle数据库中有上百个数据文件的大小变为0kb。数据库的大小缩水了80%以上。 取出&并分析oracle数据库的控制文件。重组存储位图信息,重新导出控制文件中记录的数据文件,发现这些文件的大小依然为0kb。
|
1月前
|
存储 Oracle 关系型数据库
服务器数据恢复—华为S5300存储Oracle数据库恢复案例
服务器存储数据恢复环境: 华为S5300存储中有12块FC硬盘,其中11块硬盘作为数据盘组建了一组RAID5阵列,剩下的1块硬盘作为热备盘使用。基于RAID的LUN分配给linux操作系统使用,存放的数据主要是Oracle数据库。 服务器存储故障: RAID5阵列中1块硬盘出现故障离线,热备盘自动激活开始同步数据,在同步数据的过程中又一块硬盘离线,RAID5阵列瘫痪,上层LUN无法使用。
|
15天前
|
存储 Oracle 关系型数据库
数据库传奇:MySQL创世之父的两千金My、Maria
《数据库传奇:MySQL创世之父的两千金My、Maria》介绍了MySQL的发展历程及其分支MariaDB。MySQL由Michael Widenius等人于1994年创建,现归Oracle所有,广泛应用于阿里巴巴、腾讯等企业。2009年,Widenius因担心Oracle收购影响MySQL的开源性,创建了MariaDB,提供额外功能和改进。维基百科、Google等已逐步替换为MariaDB,以确保更好的性能和社区支持。掌握MariaDB作为备用方案,对未来发展至关重要。
39 3
|
15天前
|
安全 关系型数据库 MySQL
MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!
《MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!》介绍了MySQL中的三种关键日志:二进制日志(Binary Log)、重做日志(Redo Log)和撤销日志(Undo Log)。这些日志确保了数据库的ACID特性,即原子性、一致性、隔离性和持久性。Redo Log记录数据页的物理修改,保证事务持久性;Undo Log记录事务的逆操作,支持回滚和多版本并发控制(MVCC)。文章还详细对比了InnoDB和MyISAM存储引擎在事务支持、锁定机制、并发性等方面的差异,强调了InnoDB在高并发和事务处理中的优势。通过这些机制,MySQL能够在事务执行、崩溃和恢复过程中保持
42 3
|
15天前
|
SQL 关系型数据库 MySQL
数据库灾难应对:MySQL误删除数据的救赎之道,技巧get起来!之binlog
《数据库灾难应对:MySQL误删除数据的救赎之道,技巧get起来!之binlog》介绍了如何利用MySQL的二进制日志(Binlog)恢复误删除的数据。主要内容包括: 1. **启用二进制日志**:在`my.cnf`中配置`log-bin`并重启MySQL服务。 2. **查看二进制日志文件**:使用`SHOW VARIABLES LIKE &#39;log_%&#39;;`和`SHOW MASTER STATUS;`命令获取当前日志文件及位置。 3. **创建数据备份**:确保在恢复前已有备份,以防意外。 4. **导出二进制日志为SQL语句**:使用`mysqlbinlog`
57 2
|
28天前
|
关系型数据库 MySQL 数据库
Python处理数据库:MySQL与SQLite详解 | python小知识
本文详细介绍了如何使用Python操作MySQL和SQLite数据库,包括安装必要的库、连接数据库、执行增删改查等基本操作,适合初学者快速上手。
190 15
|
22天前
|
SQL 关系型数据库 MySQL
数据库数据恢复—Mysql数据库表记录丢失的数据恢复方案
Mysql数据库故障: Mysql数据库表记录丢失。 Mysql数据库故障表现: 1、Mysql数据库表中无任何数据或只有部分数据。 2、客户端无法查询到完整的信息。
|
28天前
|
关系型数据库 MySQL 数据库
数据库数据恢复—MYSQL数据库文件损坏的数据恢复案例
mysql数据库文件ibdata1、MYI、MYD损坏。 故障表现:1、数据库无法进行查询等操作;2、使用mysqlcheck和myisamchk无法修复数据库。

热门文章

最新文章

推荐镜像

更多