Oracle LEVEL 函数练习(HR 模式 employees 表)
LEVEL 是 Oracle 伪列 ,专门用于生成层级数据 / 递归查询 ,配合 CONNECT BY 实现树形结构遍历,最适合 employees 表的员工 - 经理层级关系。
先确认环境:使用 Oracle 自带 HR 用户,核心表 employees 字段:
employee_id:员工 IDfirst_name || last_name:员工姓名manager_id:直属经理 ID(顶层总裁为 NULL)job_id:职位
基础概念
LEVEL:表示层级数(根节点 = 1,子节点 = 2,孙节点 = 3...)CONNECT BY PRIOR 子字段 = 父字段:定义层级关系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 排序字段;
总结
LEVEL是层级编号,根节点固定为 1START WITH定义从哪个节点开始遍历CONNECT BY PRIOR定义父子关联关系- 适合:组织架构、树形分类、递归遍历场景