Oracle LEVEL 函数练习(HR 模式 employees 表)

Oracle LEVEL 函数练习(HR 模式 employees 表)

LEVEL 是 Oracle 伪列 ,专门用于生成层级数据 / 递归查询 ,配合 CONNECT BY 实现树形结构遍历,最适合 employees 表的员工 - 经理层级关系

先确认环境:使用 Oracle 自带 HR 用户,核心表 employees 字段:

  • employee_id:员工 ID
  • first_name || last_name:员工姓名
  • manager_id:直属经理 ID(顶层总裁为 NULL)
  • job_id:职位

基础概念

  1. LEVEL:表示层级数(根节点 = 1,子节点 = 2,孙节点 = 3...)
  2. CONNECT BY PRIOR 子字段 = 父字段:定义层级关系
  3. START WITH:指定根节点(顶层数据)

练习 1:基础层级查询(员工层级树)

需求

查询所有员工的层级、姓名、经理 ID,从总裁开始展示完整组织架构。

复制代码
-- 员工层级树查询
SELECT
  LEVEL AS 层级,
  employee_id AS 员工ID,
  first_name || ' ' || last_name AS 员工姓名,
  manager_id AS 经理ID
FROM hr.employees
-- 根节点:经理ID为空(总裁)
START WITH manager_id IS NULL
-- 层级关系:当前员工的经理ID = 上级员工的员工ID
CONNECT BY PRIOR employee_id = manager_id
-- 按层级排序
ORDER SIBLINGS BY employee_id;

关键说明

  • LEVEL 自动生成 1、2、3... 代表组织层级
  • PRIOR employee_id上一级的员工 ID(经理)
  • 根节点:manager_id IS NULL(公司总裁)

练习 2:格式化层级展示(缩进显示树形结构)

需求

用空格缩进直观展示树形层级,更易读。

复制代码
-- 格式化树形展示
SELECT
  LEVEL AS 层级,
  -- 用 LEVEL-1 个空格缩进,模拟树形
  LPAD(' ', 2 * (LEVEL - 1)) || first_name || ' ' || last_name AS 树形员工姓名,
  employee_id,
  manager_id
FROM hr.employees
START WITH manager_id IS NULL
CONNECT BY PRIOR employee_id = manager_id
ORDER SIBLINGS BY employee_id;

练习 3:筛选指定层级(仅查询 2 级经理)

需求

只查询第 2 层级的员工(直接向总裁汇报的经理)。

复制代码
-- 筛选指定层级
SELECT
  LEVEL AS 层级,
  employee_id,
  first_name || ' ' || last_name AS 员工姓名,
  job_id
FROM hr.employees
WHERE LEVEL = 2  -- 只保留2级员工
START WITH manager_id IS NULL
CONNECT BY PRIOR employee_id = manager_id;

练习 4:查询指定员工的所有下属

需求

查询员工 King(员工 ID 100)的所有下级员工(包含子、孙层级)。

复制代码
-- 查询指定员工的全部下属
SELECT
  LEVEL AS 下属层级,
  employee_id,
  first_name || ' ' || last_name AS 下属姓名,
  manager_id
FROM hr.employees
-- 根节点:King
START WITH employee_id = 100
-- 下属关系:当前员工的经理ID = 上级员工ID
CONNECT BY PRIOR employee_id = manager_id;

练习 5:查询指定员工的所有上级(向上递归)

需求

查询员工 Austin(员工 ID 105)的所有经理(从直属到总裁)。

复制代码
-- 向上递归查询所有上级
SELECT
  LEVEL AS 上级层级,
  employee_id,
  first_name || ' ' || last_name AS 上级姓名
FROM hr.employees
-- 根节点:Austin
START WITH employee_id = 105
-- 向上关联:当前员工的经理ID = 上级员工的ID
CONNECT BY PRIOR manager_id = employee_id;

练习 6:统计每层级员工数量

需求

统计公司每个层级有多少名员工。

复制代码
-- 按层级统计人数
SELECT
  LEVEL AS 层级,
  COUNT(*) AS 员工数量
FROM hr.employees
START WITH manager_id IS NULL
CONNECT BY PRIOR employee_id = manager_id
GROUP BY LEVEL
ORDER BY 层级;

练习 7:CONNECT_BY_ROOT / CONNECT_BY_ISLEAF 进阶函数

1. CONNECT_BY_ROOT:获取根节点(顶层总裁)

复制代码
-- 每个员工 + 对应的顶层总裁
SELECT
  LEVEL AS 层级,
  first_name || ' ' || last_name AS 员工姓名,
  -- 根节点(顶层总裁)姓名
  CONNECT_BY_ROOT (first_name || ' ' || last_name) AS 顶层总裁
FROM hr.employees
START WITH manager_id IS NULL
CONNECT BY PRIOR employee_id = manager_id;

2. CONNECT_BY_ISLEAF:判断是否为叶子节点(无下属)

复制代码
-- 标记是否为最底层员工(无下属)
SELECT
  employee_id,
  first_name || ' ' || last_name AS 员工姓名,
  -- 1=叶子节点(无下属),0=非叶子(有下属)
  CONNECT_BY_ISLEAF AS 是否底层员工
FROM hr.employees
START WITH manager_id IS NULL
CONNECT BY PRIOR employee_id = manager_id;

核心语法总结

复制代码
SELECT LEVEL, 字段...
FROM 表
START WITH 根节点条件
CONNECT BY PRIOR 子节点字段 = 父节点字段
ORDER SIBLINGS BY 排序字段;

总结

  1. LEVEL 是层级编号,根节点固定为 1
  2. START WITH 定义从哪个节点开始遍历
  3. CONNECT BY PRIOR 定义父子关联关系
  4. 适合:组织架构、树形分类、递归遍历场景
相关推荐
2301_815901971 小时前
C#怎么使用协变和逆变 C#泛型中的in和out关键字协变逆变是什么意思怎么用【语法】
jvm·数据库·python
m0_463672201 小时前
SQL优化SQL关联查询中的排序字段_减少临时空间占用与内存开销
jvm·数据库·python
我叫Double1 小时前
简易版的EINO基于redis库的向量搜索项目v2
数据库·redis·bootstrap
iuvtsrt1 小时前
存储过程如何处理海量数据的批处理_循环提交与分段LIMIT结合
jvm·数据库·python
yexuhgu1 小时前
SQL如何检查字符串是否存在:INSTR与LOCATE函数使用
jvm·数据库·python
2301_783848651 小时前
SQL如何用SQL子查询实现关联报表生成_嵌套逻辑关联多表
jvm·数据库·python
techdashen2 小时前
dial9:给 Tokio 装上“飞行记录仪“
java·数据库·redis
2501_901006472 小时前
Golang怎么用gRPC Gateway_Golang gRPC Gateway教程【经典】
jvm·数据库·python
2501_901200532 小时前
golang如何实现错误预算Error Budget计算_golang错误预算Error Budget计算实现实战
jvm·数据库·python
2401_867623982 小时前
如何解决OUI图形界面无法调用_xhost与DISPLAY变量设置
jvm·数据库·python