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

Oracle — PL-SQL

介绍

     Oracle PL/SQL是专为Oracle数据库设计的过程化编程语言,深度融合SQL语句与结构化编程逻辑,旨在高效处理复杂数据操作与业务规则。其核心特征为“块结构”,程序由声明、执行、异常处理三部分组成,支持模块化开发,显著提升代码复用性和可维护性。PL/SQL通过预编译机制将代码块整体发送至数据库执行,大幅减少网络交互频次,尤其擅长批量数据处理,可借助FORALL、BULK COLLECT等特性优化事务性能。开发者可创建存储过程、函数、触发器及程序包,将业务逻辑封装于数据库层,实现数据计算下沉,保障事务一致性与安全性。异常处理框架支持自定义错误捕获与响应,增强程序健壮性。游标机制提供灵活的数据逐行处理能力,动态SQL则支持运行时语句构造,适应复杂逻辑场景。随着版本迭代,PL/SQL持续集成JSON解析、面向对象编程等现代特性,并与Java、Python等语言深度互通,巩固其在企业级应用开发中的地位,成为Oracle生态中处理高并发事务、构建金融级系统的关键技术栈。


PL/SQL 块结构

Oracle PL/SQL 块是程序基本单元,包含声明(DECLARE)、执行(BEGIN-END)、异常处理(EXCEPTION)三部分,支持变量定义、逻辑控制及错误处理,用于封装数据库操作和业务逻辑。匿名块可直接执行,存储过程等具名块可重复调用。

PL/SQL采用块结构,分为:

  1. 声明部分(DECLARE):定义变量、游标、异常等(可选)。
  2. 执行部分(BEGIN ... END):包含主要逻辑代码。
  3. 异常处理(EXCEPTION):处理运行时错误(可选)。
DECLAREv_name VARCHAR2(50) := 'Alice';
BEGINDBMS_OUTPUT.PUT_LINE('Hello, ' || v_name);
EXCEPTIONWHEN OTHERS THENDBMS_OUTPUT.PUT_LINE('Error occurred');
END;

变量与数据类型

Oracle PL/SQL变量用于存储数据,需声明数据类型,包括标量(如NUMBER、VARCHAR2)、复合(记录、集合)、引用及LOB类型,支持%TYPE继承字段类型,确保数据一致性与灵活性。

  • 标量类型:NUMBER, VARCHAR2, DATE, BOOLEAN等。
  • 复合类型:
    • 记录(RECORD):结构体类型。
TYPE t_emp IS RECORD (id NUMBER, name VARCHAR2(100));
  • 集合:包括关联数组(INDEX BY)、嵌套表(TABLE)、可变数组(VARRAY)。
  • 引用类型:%TYPE(字段类型)和%ROWTYPE(整行类型)。
  v_emp_id employees.employee_id%TYPE; -- 引用表字段类型v_emp employees%ROWTYPE; -- 引用整行结构

流程控制

Oracle PL/SQL流程控制通过条件语句(IF/CASE)、循环(FOR/WHILE/LOOP)及顺序控制(GOTO/NULL)管理代码执行逻辑,实现灵活业务处理与逻辑分支。

条件语句:

  IF condition THEN ... ELSIF ... ELSE ... END IF;CASE WHEN ... THEN ... ELSE ... END CASE;

循环:

  1. 基本循环:LOOP ... EXIT WHEN ... END LOOP;
  2. WHILE循环:WHILE condition LOOP ... END LOOP;
  3. FOR循环:FOR i IN 1..10 LOOP ... END LOOP;

游标(Cursors)

Oracle PL/SQL游标用于逐行处理查询结果集,分显式(手动声明、打开、提取、关闭)和隐式(自动管理)两种,支持循环遍历数据,实现多行记录的精确操作与复杂业务逻辑处理。

  • 显式游标:
DECLARECURSOR c_emp IS SELECT * FROM employees;v_emp employees%ROWTYPE;BEGINOPEN c_emp;LOOPFETCH c_emp INTO v_emp;EXIT WHEN c_emp%NOTFOUND;-- 处理数据END LOOP;CLOSE c_emp;END;
  • 隐式游标:自动处理SELECT INTO或DML语句。
  • FOR循环游标:
  FOR emp_rec IN (SELECT * FROM employees) LOOPDBMS_OUTPUT.PUT_LINE(emp_rec.name);END LOOP;

异常处理

  1. 预定义异常:如NO_DATA_FOUND, TOO_MANY_ROWS。
  2. 自定义异常:
  DECLAREe_custom EXCEPTION;PRAGMA EXCEPTION_INIT(e_custom, -20001);BEGINRAISE e_custom;EXCEPTIONWHEN e_custom THENDBMS_OUTPUT.PUT_LINE('自定义错误');END;

存储过程与函数

Oracle PL/SQL存储过程(PROCEDURE)封装数据库操作,无返回值;函数(FUNCTION)返回计算结果,可在SQL中调用。两者均支持参数传递,提升代码复用性、模块化及执行效率。

  • 存储过程:
  CREATE OR REPLACE PROCEDURE proc_name (p_param IN NUMBER) ISBEGIN-- 逻辑代码END;
  • 函数(必须返回一个值):
CREATE OR REPLACE FUNCTION func_name RETURN NUMBER ISBEGINRETURN 100;END;
  • 参数模式:IN(输入,默认)、OUT(输出)、IN OUT(双向)。

动态SQL

  • EXECUTE IMMEDIATE:
  EXECUTE IMMEDIATE 'UPDATE employees SET salary = :1 WHERE id = :2' USING 5000, 101;
  • DBMS_SQL包:处理复杂动态SQL。

事务控制

  1. 显式提交:COMMIT;
  2. 回滚:ROLLBACK;或回滚到保存点:ROLLBACK TO sp1;
  3. 保存点:SAVEPOINT sp1;

集合类型

  • 关联数组:
  TYPE t_dict IS TABLE OF VARCHAR2(20) INDEX BY PLS_INTEGER;
  • 嵌套表:
TYPE t_list IS TABLE OF NUMBER;
  • 可变数组(VARRAY):固定大小的数组。

触发器(Triggers)

  1. 行级触发器(FOR EACH ROW)可访问:NEW和:OLD。
  2. 事件:BEFORE/AFTER INSERT/UPDATE/DELETE。
CREATE TRIGGER trg_audit
BEFORE UPDATE ON employees
FOR EACH ROW
BEGININSERT INTO audit_table VALUES (:OLD.salary, :NEW.salary);
END;

注意事项

  1. 异常处理中避免过度使用WHEN OTHERS,应捕获具体异常。
  2. 显式游标使用后需及时关闭。
  3. 动态SQL需防范SQL注入,优先使用绑定变量(USING子句)。
  4. 事务控制需谨慎,存储过程内通常不自动提交,由调用者决定。

总结

     Oracle PL/SQL是Oracle数据库专用的过程化编程语言,深度融合SQL的数据处理能力与结构化编程特性。它以块(BLOCK)为基本单元,每个块由声明部分(DECLARE)、执行部分(BEGIN-END)和异常处理(EXCEPTION)构成,支持模块化开发,提升代码可读性与复用性。PL/SQL通过变量、条件分支(IF/CASE)、循环(LOOP/WHILE/FOR)等语法实现复杂逻辑控制,并允许开发者创建存储过程、函数、包(Package)及触发器(Trigger),将业务逻辑封装在数据库层,减少网络交互,提高执行效率。

     其异常处理机制通过预定义和自定义异常捕获错误,确保程序健壮性。游标(显式/隐式)支持逐行处理查询结果集,而集合类型(关联数组、嵌套表等)可高效操作批量数据。动态SQL技术(如EXECUTE IMMEDIATE)赋予SQL语句运行时动态构建的能力,适应灵活场景需求。PL/SQL与SQL无缝集成,支持直接在代码中嵌入DML、事务控制语句,同时通过BULK COLLECT、FORALL等特性优化批量操作性能。随着版本迭代,PL/SQL持续增强对JSON、云计算的支持,并强化调试工具(如DBMS_OUTPUT、UTL_FILE),成为企业级数据处理、事务管理和自动化任务的核心工具,广泛应用于金融、电信等领域的高性能数据库系统中。

http://www.xdnf.cn/news/6103.html

相关文章:

  • JT/T 808 各版本协议字段级别对比与解析适配建议
  • NACOS基于长链接的⼀致性模型
  • 将navicat与parcharm链接
  • 2025年中国DevOps工具选型指南:主流平台能力横向对比
  • Go语言空白导入的作用与用途
  • 【SSL部署与优化​】​​如何为网站启用HTTPS:从Let‘s Encrypt免费证书到Nginx配置​​
  • 城市生命线综合管控系统解决方案-守护城市生命线安全
  • AWS CloudTrail日志跟踪启用
  • 【计算机视觉】OpenCV实战项目:GraspPicture 项目深度解析:基于图像分割的抓取点检测系统
  • 学习51单片机01(安装开发环境)
  • 机器学习基础课程-6-课程实验
  • 精益数据分析(57/126):创业移情阶段的核心要点与实践方法
  • 前端3D动画库
  • 《隐私计算:数据安全与隐私保护的新希望》
  • Spring的Validation,这是一套基于注解的权限校验框架
  • 使用libUSB-win32的简单读写例程参考
  • zookeeper本地部署
  • 存储扇区分配表:NAND Flash与SD NAND(贴片式SD卡)的架构差异
  • spark数据压缩
  • Linux动态库与静态库
  • 通用软件项目技术报告 - 导读IV(终)
  • leetcode二叉树相关题目复习(C语言版)
  • 【ROS】将Qt的Pro工程转换到ROS2的colcon
  • java基础:异常体系
  • 006-nlohmann/json 结构转换-C++开源库108杰
  • # 深度剖析LLM的“大脑”:单层Transformer的思考模式探索
  • 青少年编程与数学 02-019 Rust 编程基础 11课题、类型系统
  • GAN简读
  • npm install 报错
  • CS4334:一款高性能的立体声音频数模转换器