百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 优雅编程 > 正文

SQL再深入之分层查询(sql查询分为)

sinye56 2024-10-08 16:31 7 浏览 0 评论

【十八】ORACLE分层查询

18.1 树的遍历

分层查询是select 语句的扩展,目的是迅速找出表中[父列--子列]的隶属关系。首先我们应该熟悉一下树的遍历方法:

ORACLE是一个关系数据库管理系统,在某些表中的数据还呈现出树型结构的联系。例如,在EMP表中含有雇员编号(EMPNO)和经理(MGR)两列,这两列反映出来的就是雇员之间领导和被领导的关系,这种关系就是一种树结构。

树的遍历有两个方向:

top--down 自上而下

即父亲找儿子,一个父亲可能有几个儿子,一个儿子可能有几个孙子,遍历不能丢了儿子,顺序以左为先。

down--top 自底向上

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

18.2 分层查询语法

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

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

18.3.1 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。

18.3.2 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

仔细看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

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

18.3 定义起始节点

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

例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.

18.4 使用LEVEL伪列

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

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

例5

1)使用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)只查看第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

SQL> select lnum,avg(sal) avgsal from (select level LNUM ,ename,mgr,sal from emp connect by prior empno=mgr start with ename='KING') group by lnum having lnum=2;

LNUM AVGSAL

---------- ----------

2 2758.33333

18.5 节点和分支的裁剪

在对树结构进行查询时,可以去掉表中的某些行,也可以剪掉树中的一个分支,使用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 子句是限定树型结构中的整个分支,既要剪除分支上的单个节点,也要剪除其后代节点(自顶向下检索时)或前辈节点(自底向顶检索时)。

例.显示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'

这个查询结果就除了剪去单个节点SCOTT外,还将SCOTT的子节点ADAMS剪掉,即把SCOTT这个分支剪掉了,当然WHERE子句可以和CONNECT BY子句联合使用,这样能够同时剪掉单个节点和树中的某个分支。

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

好了,SQL这块到这就全部结束了,后续计划介绍ORACLE的体系结构部分内容,欢迎大家点赞关注,相互学习[呲牙]!


the end !!!

@jackman 共筑美好!

相关推荐

linux 查看当前应用内存状况,以及内存参数含义

1、查看进程号ps-ef|greptomcat2、查看当前内存分配,200ms打印一次jstat-gc进程号2001jstat-gc344802001S0CS1C...

如何显示 Linux 系统上的可用内存?这几个命令很好用!

在Linux系统中,了解可用内存是优化系统性能、故障排查以及资源管理的重要一环。本文将详细介绍如何在Linux系统上显示可用内存,包括多种方法和工具的使用。在讨论可用内存之前,我们需要了解一些...

Linux 下查看内存使用情况方法总结

Q:我想监视Linux系统的内存使用情况,在Linux下有哪些视图或者命令行工具可用呢?在做Linux系统优化的时候,物理内存是其中最重要的一方面。自然的,Linux也提供了非常多的方法来监控宝贵的内...

2、linux命令-用户管理

linux命令-用户管理用户切换[root@eric~]#sueric#切换到用户eric[eric@ericroot]$[eric@ericroot]$su#切换到rootPas...

Centos 7 进入单用户模式详解

1、开机在启动菜单按e进入编辑模式找到linux16行,在最后添加init=/bin/sh编辑完后,按ctrl+x退出2、进单用户模式后,使用passwd修改密码,提示以下错误:passwd:Aut...

每日一个Linux命令解析——newusers

newusers:在Linux系统中,newusers是一个用于批量创建用户的命令。它从一个文件中读取多行用户信息,每行描述一个用户的详细信息,并根据这些信息创建多个用户或对现有用户进行批量修改。一...

openEuler操作系统管理员指南:管理用户与用户组

在Linux中,每个普通用户都有一个账户,包括用户名、密码和主目录等信息。除此之外,还有一些系统本身创建的特殊用户,它们具有特殊的意义,其中最重要的是管理员账户,默认用户名是root。同时Linux也...

Linux用户管理

1、用户信息文件/etc/passwdroot:x:0:0:root:/root:/bin/bash第一列:用户名第二列:密码位第三列:用户ID0超级用户UID。如果用户UID...

centos7基础-用户、组、权限管理

用户和组(1)用户、组、家目录的概念linux系统支持多用户,除了管理员,其他用户一般不应该使用root,而是应该向管理员申请一个账号。组类似于角色,系统可以通过组对有共性的用户进行统一管理。每个用户...

LINUX基础 ----------组及用户的概念

在Linux中,用户和组都是非常重要的概念,可以控制文件访问权限和资源的管理。用户是标识一个进程、应用程序或系统管理员的账号,Linux中每个用户用一个用户ID(UID)来标识。对于一个...

从零入门Linux(四)用户与权限管理

在Linux系统中,用户和权限管理是系统安全的重要组成部分。通过合理配置用户和权限,可以确保系统的安全性和资源的合理分配。以下是一些与用户和权限管理相关的常用命令和概念。1.用户管理1.1添加...

如何在 Linux 中管理用户?

在Linux系统中,用户是系统资源的主要使用者,每个用户都有一个唯一的标识符(用户ID)。为了更好地组织和管理用户,Linux还引入了用户组的概念。用户组是用户的集合,有助于更有效地分配权限和资...

在 Linux 中将用户添加到特定组的四种方法

在Linux多用户操作系统中,用户组管理是系统安全架构的基石。通过合理的组权限分配,管理员可以实现:精确控制文件访问权限(chmod775project/)简化批量用户权限管理(setfacl-...

我不是网管 - 如何在Ubuntu Linux下创建sudo用户

Sudo用户是Linux系统的普通用户,具有一定的管理权限,可以对系统执行管理任务。在Linux中,root是超级用户,拥有完全的管理权限,但不建议将root凭证授予其他用户或作为r...

Linux创建普通用户,为密钥方式登录做准备

Hi,我是聪慧苹果8,就是江湖上人见人爱、花见花开,土到掉榨的Linux爱好者,一起学习吧!上一篇关于SSH安全加固的文字,有网友点评通过密钥登录更加安全,先创建一个普通用户,拒绝直接使用密码登录,这...

取消回复欢迎 发表评论: