ORACLE分层查询start with&connect by

分层查询是select 语句的扩展,目的是迅速找出表中列-列的隶属关系。

19.1 树的遍历

ORACLE是一个关系数据库管理系统,它用表的形式组织数据,在某些表中的数据还呈现出树型结构的联系。例如,我们现在讨论雇员信息表EMP,其中含有雇员编号(EMPNO)和经理(MGR)两列,通过这两列反映出来的就是雇员之间领导和被领导的关系。他们之间的这种关系就是一种树结构。

图1.1 EMP表树结构图

7839
                    |
      ------------------------------------------------------------------------------------------
        |                    |                   |
      7566                                                7698                        7782
  --------------                          ---------------------------------------------------           -----  
  |        |        |      |        |              |                 |             |
  7788      7902        7499    7521    7654    7844    7900    7943    
 ------          ------
  |                 |
  7876       7369

树的遍历有两个方向

top--down 自上而下
即父亲找儿子,一个父亲可能有几个儿子,一个儿子可能有几个孙子,遍历不能丢了儿子,顺序以左为先。

down--top 自底向上
即儿子找父亲,一个儿子只能有一个父亲,所以顺序应该是:孙子->儿子-->父亲-->爷爷。

19.2 CONNECT BY 和 START WITH

在SELECT命令中使用CONNECT BY 和 START WITH 子句可以查询表中的树型结构关系。其命令格式如下:

SELECT ...
CONNECT BY {PRIOR 列名1=列名2|列名1=PRIOR 列名2}
[START WITH];

19.3 关于CONNECT BY子句

理解CONNECT BY PRIOR 子句至关重要,它确定了树的检索方向: 是top --> down(父-->子)还是down --> top(子-->父)。

在分层表中,表的父列与子列是确定的(身份固定),如:在emp表中empno是子列(下级), mgr是父列(上级)。

PRIOR关键字就像一个箭头("-->"),

connect by prior empno = mgr
connect by mgr = prior empno

两句语法等同,都是说mgr(父)--> empno(子),因此树的检索方向是top --> down。

connect by empno = prior mgr
connect by prior mgr = empno

两句语法等同,都是说empno(子)--> mgr(父),因此树的检索方向是down --> top。

19.4 START WITH 子句为可选项,用来标识哪个节点作为查找树型结构的根节点。若该子句被省略,则表示所有满足查询条件的行作为根节点(每一行都会成为一个树根)。

例1 以树结构方式显示EMP表的数据。

SQL>select empno,ename,mgr from emp
connect by prior empno=mgr
start with empno=7839
/

EMPNO ENAME                 MGR
---------- ---------- ----------
      7839 KING
      7566 JONES                7839
      7788 SCOTT                7566
      7876 ADAMS               7788
      7902 FORD                 7566
      7369 SMITH                7902
      7698 BLAKE                7839
      7499 ALLEN                7698
      7521 WARD                 7698
      7654 MARTIN               7698
      7844 TURNER               7698
      7900 JAMES                7698
      7782 CLARK                7839
      7934 MILLER               7782

14 rows selected.

仔细看empno这一列输出的顺序,就是上图树状结构每一条分支(从根节点开始)的结构。

例2 从SMITH节点开始自底向上查找EMP的树结构。

SQL>select empno,ename,mgr
from emp
connect by empno=prior mgr
start with empno=7369
/

EMPNO ENAME             MGR
---------- ---------- ----------
      7369     SMITH            7902
      7902     FORD              7566
      7566     JONES            7839
      7839     KING

SQL>

在这种自底向上的查找过程中,只有树中的一枝被显示。

19.5  定义查找起始节点

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

例3 查找7566(JONES)直接或间接领导的所有雇员信息。

SQL>SELECT EMPNO,ENAME,MGR
FROM EMP
CONNECT BY PRIOR EMPNO=MGR
START WITH EMPNO=7566
/

EMPNO ENAME             MGR
---------- ---------- ----------
      7566     JONES            7839
      7788     SCOTT            7566
      7876     ADAMS          7788
      7902     FORD             7566
      7369     SMITH            7902

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

例4 查找由FORD和BLAKE 领导的所有雇员的信息。

SQL>SELECT EMPNO,ENAME,MGR
FROM EMP
CONNECT BY PRIOR EMPNO=MGR
START WITH ENAME IN ('FORD','BLAKE')
/

EMPNO ENAME             MGR
---------- ---------- ----------
      7902     FORD             7566
      7369     SMITH            7902
      7698     BLAKE            7839
      7499     ALLEN            7698
      7521     WARD            7698
      7654     MARTIN         7698
      7844     TURNER         7698
      7900     JAMES            7698

8 rows selected.

19.6 使用LEVEL

在查询中,可以使用伪列LEVEL显示每行数据的有关层次。LEVEL将返回树型结构中当前节点的层次。

伪列LEVEL为数值型,可以在SELECT 命令中用于各种计算。

例5 使用LEVEL改变查询结果的显示形式。

SQL> COLUMN LEVEL FORMAT A20

SQL> SELECT LPAD(LEVEL,LEVEL*3,' ')
as "LEVEL",EMPNO,ENAME,MGR
FROM EMP
CONNECT BY PRIOR EMPNO=MGR
START WITH ENAME='KING'
/

LEVEL                     EMPNO ENAME             MGR
-------------------- ---------- ---------- ----------
  1                        7839 KING
     2                     7566 JONES                7839
        3                  7788 SCOTT                7566
           4               7876 ADAMS               7788
        3                  7902 FORD                 7566
           4               7369 SMITH                7902
     2                     7698 BLAKE                7839
        3                  7499 ALLEN                7698
        3                  7521 WARD                 7698
        3                  7654 MARTIN              7698
        3                  7844 TURNER               7698
        3                  7900 JAMES                7698
     2                     7782 CLARK                7839
        3                  7934 MILLER               7782

14 rows selected.

在SELECT使用了函数LPAD,该函数表示以LEVEL*3个空格进行填充,由于不同行处于不同的节点位置,具有不同的LEVEL值,因此填充的空格数将根据各自的层号确定,空格再与层号拼接,结果显示出这种层次关系。

只查看第2层的员工信息:

SQL> select t1.* from (select level LNUM ,ename,mgr from emp connect by prior empno=mgr start with ename='KING') t1 where LNUM=2;

LNUM ENAME          MGR
---------- ---------- ----------
         2     JONES            7839
         2     BLAKE            7839
         2     CLARK            7839

19.7 节点和分支的裁剪

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

SQL>SELECT LPAD(LEVEL,LEVEL*3,' ')
as "LEVEL",EMPNO,ENAME,MGR
FROM EMP
WHERE ENAME<>'SCOTT'
CONNECT BY PRIOR EMPNO=MGR
START WITH ENAME='KING'
/

LEVEL                     EMPNO ENAME             MGR
-------------------- ---------- ---------- ----------
  1                        7839     KING
     2                     7566     JONES            7839
           4               7876     ADAMS          7788
        3                  7902     FORD             7566
           4               7369    SMITH            7902
     2                     7698     BLAKE            7839
        3                  7499     ALLEN            7698
        3                  7521     WARD            7698
        3                  7654     MARTIN         7698
        3                  7844     TURNER          7698
        3                  7900     JAMES            7698
     2                     7782     CLARK            7839
        3                  7934     MILLER           7782

13 rows selected.

在这个查询中,仅剪去了树中单个节点SCOTT。若希望剪去树结构中的某个分支,则要用CONNECT BY 子句。CONNECT BY 子句是限定树型结构中的整个分支,既要剪除分支上的单个节点,也要剪除其后代节点(自顶向下检索时)或前辈节点(自底向顶检索时)。

例8.显示KING领导下的全体雇员信息,除去SCOTT领导的一支。

SQL>SELECT LPAD(LEVEL,LEVEL*3,' ')
as "LEVEL",EMPNO,ENAME,MGR
FROM EMP
CONNECT BY PRIOR EMPNO=MGR
AND ENAME!='SCOTT'
START WITH ENAME='KING'
/

LEVEL                     EMPNO ENAME             MGR
-------------------- ---------- ---------- ----------
  1                        7839     KING
     2                     7566     JONES            7839
        3                  7902     FORD             7566
           4               7369     SMITH            7902
     2                     7698     BLAKE            7839
        3                  7499     ALLEN            7698
        3                  7521     WARD            7698
        3                  7654     MARTIN         7698
        3                  7844     TURNER         7698
        3                  7900     JAMES            7698
     2                     7782     CLARK            7839
        3                  7934     MILLER           7782

12 rows selected

这个查询结果就除了剪去单个节点SCOTT外,还将SCOTT的子节点ADAMS剪掉,即把SCOTT这个分支剪掉了。

当然WHERE子句可以和CONNECT BY子句联合使用,这样能够同时剪掉单个节点和树中的某个分支。

例9.显示KING领导全体雇员信息,除去雇员SCOTT,以及BLAKE领导的一支。

(这个留给大家实践吧:)

在使用SELECT 语句来报告树结构报表时应当注意,CONNECT BY子句不能作用于出现在WHERE子句中的表连接。如果需要进行连接,可以先用树结构建立一个视图,再将这个视图与其他表连接,以完成所需要的查询。

oracle之分层查询的更多相关文章

  1. Oracle DBA常用查询

    Oracle DBA常用查询 –1. 查询系统所有对象select owner, object_name, object_type, created, last_ddl_time, timestamp ...

  2. oracle系统表查询

    oracle查询用户下的所有表 select * from all_tab_comments -- 查询所有用户的表,视图等select * from user_tab_comments -- 查询本 ...

  3. Oracle中分页查询语句

    Oracle分页查询语句使我们最常用的语句之一,下面就为您介绍的Oracle分页查询语句的用法,如果您对此方面感兴趣的话,不妨一看. Oracle分页查询语句基本上可以按照本文给出的格式来进行套用.O ...

  4. oracle数据库常用查询一

    oracle数据库常用查询一 sqlplus / as sysdba; 或sqlplus sys/密码 as sysdba;两者都是以sys登录.conn scott/tiger@orcl; conn ...

  5. ORACLE数据库常用查询二

    ORACLE数据库常用查询 1.查看表空间对应数据文件情况: SQL MB,AUTOEXTENSIBLE FROM DBA_DATA_FILES; TABLESPACE_NAME FILE_NAME ...

  6. Oracle的分页查询语句优化

    Oracle的分页查询语句基本上可以按照本文给出的格式来进行套用. (一)   分页查询格式: SELECT * FROM  ( SELECT A.*, ROWNUM RN  FROM (SELECT ...

  7. Oracle使用游标查询指定数据表的所有字段名称组合而成的字符串

    应用场合:参考网上查询数据表的所有字段名代码,使用游标生成指定单个表的所有字段名跟逗号组成的用于select  逗号隔开的字段名列表 from字符串等场合. 查询结果输出如下: 当前数据表TB_UD_ ...

  8. ORACLE跨数据库查询的方法

    原文地址:http://blog.csdn.net/huzhenwei/article/details/2533869 本文简述了通过创建database link实现Oracle跨数据库查询的方法 ...

  9. oracle 中对查询出来的数据进行切割、截取等操作

    oracle 中对查询出来的数据进行切割.截取等操作 最近遇到一个问题,需要把一个带有,的字符串拆分成多行.通过查询资料,这个操作需要使用以下2个关键知识: 1. REGEXP_SUBSTR函数 这个 ...

随机推荐

  1. c++排序二叉树的出现的私有函数讨论,以及二叉树的删除操作详解

    c++排序二叉树的出现的私有函数讨论, 以及二叉树的删除操作详解 标签(空格分隔): c++ 前言 我在c++学习的过程中, 最近打了一个排序二叉树的题目,题目中出现了私有函数成员,当时没有理解清楚这 ...

  2. centos go 安装 使用

    #goland 确保能ping通百度[root@z my_project]# vi /etc/resolv.conf# Generated by NetworkManagersearch locald ...

  3. 图论算法(三) 最短路SPFA算法

    我可能要退役了…… 退役之前,写一篇和我一样悲惨的算法:SPFA 最短路算法(二)SPFA算法 Part 1:SPFA算法是什么 其实呢,SPFA算法只是在天朝大陆OIers的称呼,它的正统名字叫做: ...

  4. go语言gRPC系列(三) - 使用grpc-gateway同时提供HTTP和gRPC服务

    1. gRPC提供HTTP服务 1.1 存在的意义 1.2 代码示例 1.3 使用postman尝试调用 1.4 gRPC客户端代码调用 2. 使用grpc-gateway同时提供HTTP和gRPC服 ...

  5. 【算法•日更•第三十一期】KMP算法

    ▎前言 这次要讲的HMP算法KMP算法很简单,是用于处理字符串的,之前一直以为很难,其实也不过如此(说白了就是优化一下暴力). ▎处理的问题 通常处理的问题是这样的:给定两个字符串s1和s2,其中s1 ...

  6. 轻松解除官方对博客后台设置页面3段html代码的限制:iframe alert flagcounter etc.的禁用

    今天我发现我的flagcounter不能用了 iframe也不能用了,无奈╮(╯_╰)╭ 第一反应是Ctrl U viewsource看看源码出了什么问题 结果竟然是这样, 因为页首和侧边栏的html ...

  7. graphics.h源代码下载

    graphics.h源代码下载 /*graphics.h DefinitionsforGraphicsPackage. Copyright(c)BorlandInternational1987,198 ...

  8. C++统计单词数

    [题目描述] 一般的文本编辑器都有查找单词的功能,该功能可以快速定位特定单词在文章中的位置,有的还能统计出特定单词在文章中出现的次数.现在,请你编程实现这一功能,具体要求是:给定一个单词,请你输出它在 ...

  9. 进阶6:连接查询 二、sql99语法

    #二.sql99语法/*语法: select 查询列表 from 表1 别名 [连接类型] join 表2 别名 on 连接条件 [where 筛选条件] [group by 分组] [having ...

  10. 第2章 执行SparkSQL查询

    第2章 执行SparkSQL查询 2.1 命令行查询流程 打开Spark shell 例子:查询大于30岁的用户 创建如下JSON文件,注意JSON的格式: {"name":&qu ...