当前位置: 首页 > news >正文

Oracle数据库数据编程SQL<递归函数详解>

递归函数是一种在函数体内直接或间接调用自身的函数。这种函数通过将复杂问题分解为更小的相同问题来解决特定类型的编程任务。

目录

一、递归函数基本概念

1. 递归定义

2. 递归工作原理

 二、递归函数示例

1. 经典阶乘函数

2. 斐波那契数列

3. 计算数字位数

三、递归查询应用

1. 递归WITH子句(公用表表达式)

2. 层次结构查询(员工-经理关系)

 四、递归函数设计要点

 五、递归与迭代对比

1. 递归与迭代对比

2. 迭代实现阶乘(对比) 

六、递归函数的高级应用

1. 遍历树形结构

2. 解决汉诺塔问题

七、递归的局限性及优化

1. Oracle递归深度限制

2. 尾递归优化

3. 记忆化(Memoization)优化


一、递归函数基本概念

1. 递归定义

递归函数包含两个关键部分:

  • 基准条件(Base Case):确定递归何时结束,避免无限循环

  • 递归条件(Recursive Case):将问题分解为更小的子问题并调用自身

2. 递归工作原理

  • (1)函数检查是否满足基准条件
  • (2)如果满足,返回基准值
  • (3)如果不满足,将问题分解并调用自身处理更小的子问题
  • (4)最终将所有子问题的结果组合成最终解

 二、递归函数示例

1. 经典阶乘函数

CREATE OR REPLACE FUNCTION factorial(
    p_num IN NUMBER
) RETURN NUMBER
IS
BEGIN
    -- 基准条件:0!和1!都等于1
    IF p_num <= 1 THEN
        RETURN 1;
    ELSE
        -- 递归条件:n! = n * (n-1)!
        RETURN p_num * factorial(p_num - 1);
    END IF;
END factorial;
/

-- 调用示例
SELECT factorial(5) FROM dual;  -- 返回120 (5! = 5×4×3×2×1 = 120)

2. 斐波那契数列

CREATE OR REPLACE FUNCTION fibonacci(
    p_num IN NUMBER
) RETURN NUMBER
IS
BEGIN
    -- 基准条件
    IF p_num = 0 THEN
        RETURN 0;
    ELSIF p_num = 1 THEN
        RETURN 1;
    ELSE
        -- 递归条件:F(n) = F(n-1) + F(n-2)
        RETURN fibonacci(p_num - 1) + fibonacci(p_num - 2);
    END IF;
END fibonacci;
/

-- 调用示例
SELECT fibonacci(10) FROM dual;  -- 返回55 (斐波那契数列第10项)

3. 计算数字位数

CREATE OR REPLACE FUNCTION count_digits(
    p_num IN NUMBER
) RETURN NUMBER
IS
BEGIN
    -- 基准条件:单个数字
    IF p_num BETWEEN -9 AND 9 THEN
        RETURN 1;
    ELSE
        -- 递归条件:去掉最后一位后计数+1
        RETURN 1 + count_digits(TRUNC(p_num/10));
    END IF;
END count_digits;
/

-- 调用示例
SELECT count_digits(12345) FROM dual;  -- 返回5

三、递归查询应用

 Oracle中的递归不仅限于PL/SQL函数,还可以用于SQL递归查询:

1. 递归WITH子句(公用表表达式)

-- 计算1到10的和
WITH recursive_sum(n, total) AS (
    -- 基准查询
    SELECT 1, 1 FROM dual
    
    UNION ALL
    
    -- 递归查询
    SELECT n+1, total+(n+1)
    FROM recursive_sum
    WHERE n < 10
)
SELECT * FROM recursive_sum;

/*
结果:
N  TOTAL
1     1
2     3
3     6
4    10
5    15
6    21
7    28
8    36
9    45
10   55
*/

2. 层次结构查询(员工-经理关系)

-- 查找员工的管理链
WITH emp_hierarchy AS (
    -- 基准查询:从指定员工开始
    SELECT employee_id, last_name, manager_id, 1 AS level
    FROM employees
    WHERE employee_id = 110
    
    UNION ALL
    
    -- 递归查询:向上查找经理
    SELECT e.employee_id, e.last_name, e.manager_id, h.level + 1
    FROM employees e
    JOIN emp_hierarchy h ON e.employee_id = h.manager_id
)
SELECT LPAD(' ', 2*(level-1)) || last_name AS management_chain
FROM emp_hierarchy
ORDER BY level DESC;

/*
结果:
MANAGEMENT_CHAIN
    King
      Higgins
        Gietz
*/

 四、递归函数设计要点

  1. 必须存在基准条件:确保递归能够终止

  2. 每次递归应使问题更接近基准条件:避免无限递归

  3. 注意递归深度:Oracle有递归深度限制(默认约1000层)

  4. 性能考虑:递归可能比迭代解决方案效率低

  5. 内存使用:深层递归会消耗大量栈空间

 五、递归与迭代对比

1. 递归与迭代对比

特性递归迭代
代码简洁性更简洁通常更冗长
内存使用使用调用栈,可能栈溢出通常更高效
性能函数调用开销大通常更快
适用场景问题自然递归时大多数情况
可读性数学定义清晰时更易读流程更直观

2. 迭代实现阶乘(对比) 

CREATE OR REPLACE FUNCTION factorial_iter(
    p_num IN NUMBER
) RETURN NUMBER
IS
    v_result NUMBER := 1;
BEGIN
    FOR i IN 1..p_num LOOP
        v_result := v_result * i;
    END LOOP;
    
    RETURN v_result;
END factorial_iter;
/

六、递归函数的高级应用

1. 遍历树形结构

CREATE OR REPLACE PROCEDURE print_folder_tree(
    p_folder_id IN NUMBER,
    p_level IN NUMBER DEFAULT 0
) IS
    CURSOR folder_cur IS
        SELECT folder_id, folder_name
        FROM folders
        WHERE parent_id = p_folder_id;
BEGIN
    -- 打印当前文件夹(带缩进)
    DBMS_OUTPUT.PUT_LINE(LPAD(' ', p_level*2) || 'Folder: ' || p_folder_id);
    
    -- 递归处理子文件夹
    FOR f_rec IN folder_cur LOOP
        print_folder_tree(f_rec.folder_id, p_level + 1);
    END LOOP;
END print_folder_tree;
/

2. 解决汉诺塔问题

CREATE OR REPLACE PROCEDURE hanoi(
    p_disks IN NUMBER,
    p_from IN VARCHAR2,
    p_to IN VARCHAR2,
    p_aux IN VARCHAR2
) IS
BEGIN
    IF p_disks = 1 THEN
        DBMS_OUTPUT.PUT_LINE('移动盘子 1 从 ' || p_from || ' 到 ' || p_to);
    ELSE
        -- 将n-1个盘子从起始柱移动到辅助柱
        hanoi(p_disks - 1, p_from, p_aux, p_to);
        
        -- 移动第n个盘子到目标柱
        DBMS_OUTPUT.PUT_LINE('移动盘子 ' || p_disks || ' 从 ' || p_from || ' 到 ' || p_to);
        
        -- 将n-1个盘子从辅助柱移动到目标柱
        hanoi(p_disks - 1, p_aux, p_to, p_from);
    END IF;
END hanoi;
/

-- 调用示例
EXEC hanoi(3, 'A', 'C', 'B');

七、递归的局限性及优化

1. Oracle递归深度限制

 Oracle默认递归深度限制约为1000层,可通过以下方式调整:

ALTER SESSION SET recursive_with_clause = 10000;  -- 增加递归WITH限制

对于PL/SQL递归函数,

可通过修改_recursive_with_balancing等隐藏参数调整限制(需DBA权限)

2. 尾递归优化

Oracle PL/SQL不自动执行尾递归优化,但可手动重构:

-- 普通递归
CREATE OR REPLACE FUNCTION sum_to_n(
    p_n IN NUMBER
) RETURN NUMBER
IS
BEGIN
    IF p_n <= 1 THEN
        RETURN 1;
    ELSE
        RETURN p_n + sum_to_n(p_n - 1);
    END IF;
END;
/

-- 尾递归形式(Oracle不会优化,但其他语言可能优化)
CREATE OR REPLACE FUNCTION sum_to_n_tail(
    p_n IN NUMBER,
    p_acc IN NUMBER DEFAULT 0
) RETURN NUMBER
IS
BEGIN
    IF p_n <= 0 THEN
        RETURN p_acc;
    ELSE
        RETURN sum_to_n_tail(p_n - 1, p_acc + p_n);
    END IF;
END;
/

3. 记忆化(Memoization)优化

缓存已计算结果,避免重复计算:

CREATE OR REPLACE PACKAGE fib_pkg AS
    TYPE num_array IS TABLE OF NUMBER INDEX BY PLS_INTEGER;
    g_fib_cache num_array;
    
    FUNCTION fibonacci(p_n NUMBER) RETURN NUMBER;
END fib_pkg;
/

CREATE OR REPLACE PACKAGE BODY fib_pkg AS
    FUNCTION fibonacci(p_n NUMBER) RETURN NUMBER
    IS
    BEGIN
        -- 基准条件
        IF p_n = 0 THEN
            RETURN 0;
        ELSIF p_n = 1 THEN
            RETURN 1;
        END IF;
        
        -- 检查缓存
        IF g_fib_cache.EXISTS(p_n) THEN
            RETURN g_fib_cache(p_n);
        END IF;
        
        -- 计算并缓存结果
        g_fib_cache(p_n) := fibonacci(p_n - 1) + fibonacci(p_n - 2);
        RETURN g_fib_cache(p_n);
    END;
END fib_pkg;
/

递归函数是解决分治问题和层次结构处理的强大工具,但需要谨慎设计以避免性能问题和栈溢出错误。在Oracle环境中,对于深度递归问题,有时使用迭代方法或递归WITH子句可能是更好的选择。

相关文章:

  • Linux之权限问题
  • 2025普通话考试(最新)| 普通话软件 + 真题 + 测试
  • Python第六章20:函数的参数传递和匿名函数
  • NQA 网络质量分析协议
  • 哈希表 - 两个数组的交集(集合、数组) - JS
  • 华为hcia——Datacom实验指南——配置OSPF路由
  • 【Sa-Token】学习笔记 03 - 认识Sa-Token中常见类
  • Linux线程池实现
  • [吾爱出品] 防窥助手 V1.1
  • GIC硬件
  • C++_STL之list篇
  • 衡石科技HENGSHI SENSE异构数据关联技术深度解析:揭秘5-8倍性能提升背后的“异构过滤“架构
  • 第一讲 感应加热原理与感应熔炼电炉特点
  • Java抽象类:深入理解与应用
  • 前端调试实践与案例场景
  • Redis-08.Redis常用命令-有序集合操作命令
  • Socket函数详解:完整指南
  • 用 Python 实现一个 Benchmark 工具
  • MySQL数据库精研之旅第五期:CRUD的趣味探索(上)
  • Assembly语言的嵌入式调试