Oracle 第11章:异常处理
在 Oracle PL/SQL 中,异常处理是一个重要的概念,它用于管理程序执行过程中可能发生的错误或特殊情况。异常可以是系统预定义的,也可以是由用户自定义的。
异常类型与处理机制
PL/SQL 提供了两种类型的异常:
- 预定义异常:这些是由 Oracle 数据库自动引发的标准异常,如
NO_DATA_FOUND(没有找到数据),TOO_MANY_ROWS(返回多行数据),等等。 - 用户定义异常:开发者可以根据需要定义自己的异常来处理特定的情况。
异常处理的基本结构包括 EXCEPTION 块,在这个块中可以编写处理特定异常的代码。如果在 PL/SQL 块中抛出了异常,并且该异常没有被捕获,则会导致 PL/SQL 块的执行终止。
示例代码
下面是一个使用预定义异常 NO_DATA_FOUND 的简单示例:
DECLAREv_salary NUMBER;
BEGIN-- 尝试从员工表中获取员工的工资SELECT salary INTO v_salary FROM employees WHERE employee_id = 9999;-- 如果没有找到该员工,则会抛出 NO_DATA_FOUND 异常
EXCEPTIONWHEN NO_DATA_FOUND THENdbms_output.put_line('Employee not found.');
END;
/
在这个例子中,如果尝试查询一个不存在的员工记录,SELECT ... INTO 语句将无法成功执行,并触发 NO_DATA_FOUND 异常。我们通过 WHEN NO_DATA_FOUND 子句捕获这个异常,并输出一条信息。
用户定义异常
有时候,标准异常不足以描述某些特定错误情况,这时就需要定义自己的异常。定义自定义异常的方法如下:
DECLAREe_invalid_value EXCEPTION; -- 定义一个异常PRAGMA EXCEPTION_INIT(e_invalid_value, -20001); -- 给异常赋一个错误码
BEGIN-- 检查某个条件是否满足,如果不满足则抛出自定义异常IF salary < 0 THENRAISE e_invalid_value;END IF;
EXCEPTIONWHEN e_invalid_value THENdbms_output.put_line('Invalid value for salary.');
END;
/
在这个例子中,我们定义了一个名为 e_invalid_value 的自定义异常,并给它分配了一个 Oracle 错误码 -20001。当检查到工资值为负数时,我们就手动抛出这个异常。然后在 EXCEPTION 块中处理这个异常。
注意事项
- 在实际应用中,应该尽量避免使用通用异常处理器(如
WHEN OTHERS),因为这可能会掩盖真正的错误源。 - 使用异常处理时,要确保所有的可能异常都被正确处理,以防止程序崩溃。
- 在定义自定义异常时,确保错误码的选择不会与 Oracle 预留的错误码冲突。
多个异常的处理
在复杂的业务逻辑中,可能需要处理多种不同的异常。在这种情况下,可以在 EXCEPTION 块中为每种异常提供特定的处理逻辑。下面是一个处理多个异常的例子:
DECLAREv_employee_id employees.employee_id%TYPE := 9999;v_salary employees.salary%TYPE;
BEGIN-- 尝试从员工表中获取员工的工资SELECT salary INTO v_salary FROM employees WHERE employee_id = v_employee_id;-- 如果没有找到记录,或者返回了多条记录,都会抛出相应的异常
EXCEPTIONWHEN NO_DATA_FOUND THENdbms_output.put_line('No data found for employee ID ' || v_employee_id);WHEN TOO_MANY_ROWS THENdbms_output.put_line('Too many rows returned for employee ID ' || v_employee_id);WHEN OTHERS THEN-- 记录其他所有未预见的异常dbms_output.put_line('An unexpected error occurred: ' || SQLERRM);
END;
/
在这个例子中,我们增加了对 TOO_MANY_ROWS 异常的处理,以防查询返回多条记录的情况。同时,也保留了 WHEN OTHERS 子句来捕捉任何未预见的异常,并打印出错误信息。
使用自定义异常的最佳实践
当你创建自定义异常时,应该遵循以下最佳实践:
- 明确错误消息:为自定义异常提供明确的错误消息,以便于调试。
- 异常隔离:尽可能地隔离不同类型的异常,使得每种异常都有其特定的处理逻辑。
- 使用子类型:如果你有很多相似但又略有不同的异常,考虑定义一个基异常类型,并基于此类型定义子类型。
示例:自定义异常的使用
假设我们需要确保输入的数据符合一定的格式要求,我们可以定义一系列相关的异常:
DECLARETYPE validation_exception IS EXCEPTION;PRAGMA EXCEPTION_INIT(validation_exception, -20000);e_invalid_date validation_exception;e_invalid_name validation_exception;v_date DATE;v_name VARCHAR2(100);
BEGINv_date := TO_DATE('2024-02-30', 'YYYY-MM-DD'); -- 这里会抛出异常v_name := 'John Doe'; -- 假设这里有一个验证逻辑IF v_date > SYSDATE + 365 THENRAISE e_invalid_date;END IF;IF v_name NOT LIKE '%[A-Z][a-z]*%' THENRAISE e_invalid_name;END IF;-- 如果没有异常发生,继续执行后续逻辑...
EXCEPTIONWHEN e_invalid_date THENdbms_output.put_line('The date is invalid.');WHEN e_invalid_name THENdbms_output.put_line('The name is invalid.');WHEN OTHERS THENdbms_output.put_line('Unexpected error: ' || SQLERRM);
END;
/
在这个示例中,我们定义了两个自定义异常 e_invalid_date 和 e_invalid_name,分别用于验证日期和名字的有效性。每个异常都有具体的检查逻辑,并且在 EXCEPTION 块中有对应的处理代码。
通过这种方式,你可以为应用程序添加更强大的错误检测和处理功能,同时提高代码的可读性和可维护性。
复杂应用场景中的异常处理
在更复杂的应用场景中,异常处理不仅限于简单的错误捕捉,还需要考虑异常传播、异常恢复等更高级的功能。例如,当一个事务中的一部分操作失败时,我们可能希望回滚整个事务,而不是仅仅处理单个操作的异常。
示例:事务处理与异常
假设我们正在实现一个银行转账的业务逻辑,涉及到从一个账户扣款并存入另一个账户。为了保证数据的一致性,我们需要在一个原子性的事务中完成这两个操作。如果其中一个操作失败,我们需要回滚整个事务。
DECLAREv_sender_account accounts.account_id%TYPE := '12345678';v_receiver_account accounts.account_id%TYPE := '87654321';v_amount NUMBER := 100;
BEGIN-- 开始一个新的事务SAVEPOINT start_of_transaction;-- 从发送者账户扣除金额UPDATE accounts SET balance = balance - v_amount WHERE account_id = v_sender_account RETURNING balance INTO v_sender_balance;-- 如果没有找到发送者账户或余额不足,回滚事务IF SQL%NOTFOUND OR v_sender_balance < 0 THENROLLBACK TO start_of_transaction;RAISE_APPLICATION_ERROR(-20001, 'Insufficient funds or account does not exist.');END IF;-- 向接收者账户增加金额UPDATE accounts SET balance = balance + v_amount WHERE account_id = v_receiver_account RETURNING balance INTO v_receiver_balance;-- 如果没有找到接收者账户,回滚事务IF SQL%NOTFOUND THENROLLBACK TO start_of_transaction;RAISE_APPLICATION_ERROR(-20002, 'Receiver account does not exist.');END IF;-- 如果一切顺利,提交事务COMMIT;
EXCEPTIONWHEN OTHERS THEN-- 回滚事务并记录错误信息ROLLBACK TO start_of_transaction;dbms_output.put_line('Transaction failed: ' || SQLERRM);
END;
/
在这个例子中,我们使用了 SAVEPOINT 来标记事务的开始点,并在每次更新操作之后检查是否存在异常。如果有异常发生,我们会回滚到事务的起点。只有当两个操作都成功时,才会提交事务。
异常处理与错误日志
除了处理异常本身外,记录详细的错误日志也是非常重要的一部分。当异常发生时,记录详细的错误信息可以帮助后续的故障排查和问题解决。
示例:记录错误日志
在 PL/SQL 中,可以使用 DBMS_OUTPUT.PUT_LINE 来输出简单的错误信息,但对于生产环境来说,通常会使用更专业的日志记录工具,如 DBMS_LOGMNR 或者其他日志框架来记录详细的错误信息。
BEGIN-- 尝试执行某项操作INSERT INTO transactions (id, amount) VALUES (1, 100);-- 如果插入失败,记录错误信息
EXCEPTIONWHEN DUP_VAL_ON_INDEX THENdbms_output.put_line('Duplicate value on index.');-- 在实际应用中,这里应该调用专门的日志记录过程-- LOG_ERROR('Duplicate value on index.', SQLERRM);
END;
/
在这个例子中,如果插入操作导致索引重复键异常,我们不仅输出了一个简单的错误信息,还可以记录详细的错误信息到日志文件中。
总结
通过上述讨论,我们可以看到异常处理不仅仅是简单的错误捕捉,而是涵盖了错误处理、事务管理和错误日志记录等多个方面。合理地设计和实现异常处理逻辑,可以使应用程序更加健壮和易于维护。在实际开发中,应该根据具体的应用场景和需求来选择合适的异常处理策略。
相关文章:
Oracle 第11章:异常处理
在 Oracle PL/SQL 中,异常处理是一个重要的概念,它用于管理程序执行过程中可能发生的错误或特殊情况。异常可以是系统预定义的,也可以是由用户自定义的。 异常类型与处理机制 PL/SQL 提供了两种类型的异常: 预定义异常…...
导航栏渐变色iOS
- (void)viewDidLoad {[super viewDidLoad];// 设置导航栏属性self.navigationBar.translucent NO;[self.navigationBar setTitleTextAttributes:{NSForegroundColorAttributeName : [UIColor whiteColor], NSFontAttributeName:[UIFont boldSystemFontOfSize:28]}];// 修复iO…...
mysql读写分离
一、proxysql实现mysql读写分离 二、mycat...
计算机的错误计算(一百四十二)
摘要 本节探讨 MATLAB中 附近数的正弦函数的计算精度问题。 例1. 已知 计算 与 直接贴图吧: 另外, 16位的正确值分别为 -0.3077518861551721e-8 与 0.4106402475009074e-3(ISRealsoft 提供)。 容易看出,MATLAB的…...
利用大模型辅助科研论文写作·第一期|论文写作·24-11-02
小罗碎碎念 从这期推文开始,开一个新的系列——如何利用大语言模型辅助论文写作。 我目前的推文主要都集中于分享已经发表的论文,前期背景积累到一定程度以后,我们要动手做实验然后写自己的论文。如果从头到尾,全都自己写…...
JavaScript。—关于语法基础的理解—
一、程序控制语句 JavaScript 提供了 if 、if else 和 switch 3种条件语句,条件语句也可以嵌套。 (一)、条件语句 1、单向判断 : if... (1)概述 < if >元素用于在判断该语句是否满足特定条…...
Tomcat 11 下载/安装 与基本使用
为什么要使用Tomcat? 使用Apache Tomcat的原因有很多,以下是一些主要的优点和特点: 1. 开源与免费 Tomcat是一个完全开源的项目,任何人都可以免费使用。它由Apache软件基金会维护,拥有一个活跃的社区,这…...
Linux系统时间服务——Chrony服务器
文章目录 Linux系统时间服务——Chrony服务器前言时间同步的重要性Linux系统的两种时钟系统时钟(System Clock)相关命令硬件时钟 (RTC - Real Time Clock)相关命令 Chrony介绍NTP Chronyc相关命令服务管理相关命令chronyc 基本命令时间校正和控制命令NTP…...
C# 接口(Interface)
C# 接口(Interface) 接口在C#中是一种非常重要的概念,它定义了一个约定,实现该接口的类必须遵循这个约定。接口可以包含方法、属性、事件和索引器,但不包含实现。这使得接口成为定义抽象行为的理想选择。在本文中&…...
《高频电子线路》—— 电容三端LC振荡器
文章内容来源于【中国大学MOOC 华中科技大学通信(高频)电子线路精品公开课】,此篇文章仅作为笔记分享。 电容三端LC振荡器 基本原理(考毕兹电路) 反馈电压从C2上取得,作为输入电压,形成正反馈&a…...
leetcode35.搜索插入位置
1)题目描述: 2)本题要求使用 时间复杂度O(log n)的算法,这里使用二分查找的方法,这道题本身不复杂,但是,在使用递归调用时,笔者经常把递归结束的边界搞错,这里给出几版代…...
Redis全系列学习基础篇之位图(bitmap)常用命令的解析
文章目录 描述常用命令及解析常用命令解析 应用场景统计不确定时间周期内用户登录情况思路分析实现 统计某一特定时间内活跃用户(登录一次即算活跃)的数量思路分析与实现 描述 bitmap是redis封装的用于针对位(bit)的操作,其特点是计算效率高,占用空间少,常被用来统计…...
Copilot功能
Copilot 1、简介:Copilot是由GitHub与OpenAI共同开发的一款AI编程助手,旨在帮助开发者提高工作效率,改善代码质量。 2、主要功能包括: 1.代码补全:Copilot可以在开发者编写代码时提供代码建议,包括函数、循…...
《GBDT 算法的原理推导》 11-13初始化模型 公式解析
本文是将文章《GBDT 算法的原理推导》中的公式单独拿出来做一个详细的解析,便于初学者更好的理解。 公式(11-13)是GBDT算法的第一步,它描述了如何初始化模型。公式如下: f 0 ( x ) arg min c ∑ i 1 N L ( y i , c ) f_0(x) \arg \m…...
# Easysearch 与 LLM 融合打造高效智能问答系统
LangChain通过提供统一的抽象层和丰富的工具,极大地简化了LLM应用程序的开发过程,使得开发者能够更加专注于业务逻辑。RAG技术则通过索引和检索生成两步流程,利用最新数据或私有数据作为背景信息来增强大模型的推理能力。然而,对于…...
本地可以插入表记录,生产不能插入表记录
先说解决方案: 切面没有注入容器,在切面这加上Component详情: 大致是这样一个方法,本地运行会插入数据到sys_log表,但部署到服务器上就不会插入,而服务部署三年多了,一个表一直是空的居然没人…...
11.Three.js使用indexeddb前端缓存模型优化前端加载效率
11.Three.js使用indexeddb前端缓存模型优化前端加载效率 1.简述 在使用Three.js做数字孪生应用场景时,我们常常需要用到大量模型或数据。在访问我们的数字孪生应用时,每次刷新都需要从web端进行请求大量的模型数据或其他渲染数据等等,会极大…...
功能测试:方法、流程与工具介绍
功能测试是对产品的各功能进行验证的一种测试方法,旨在确保软件以期望的方式运行并满足设计需求。以下是对功能测试的详细解释: 一、定义与目的 定义:功能测试(Functional Testing),也称为行为测试&#…...
【Orange Pi 5 Linux 5.x 内核编程】-设备驱动中的sysfs
设备驱动中的sysfs 文章目录 设备驱动中的sysfs1、sysfs介绍2、内核对象(kobject)介绍3、设备驱动中的SysFS31 在/sys中创建目录3.2 创建sysfs文件3.2.1 创建属性3.2.2 创建sysfs文件4、驱动程序实现5、驱动验证1、sysfs介绍 sysfs是内核导出的虚拟文件系统,类似于/proc。sys…...
微信小程序-全局数据共享/页面间通信
一.全局数据共享 声明全局的变量,在app.js文件里 App({//全局共享的数据globalData:{token:},//设置全局数据setToken(token){this.globalData.tokentoken}})使用 getApp() 获取全局App实例 //返回全局唯一的APP实例 const appInstancegetApp()Page({login(){con…...
OpenLayers 可视化之热力图
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...
地震勘探——干扰波识别、井中地震时距曲线特点
目录 干扰波识别反射波地震勘探的干扰波 井中地震时距曲线特点 干扰波识别 有效波:可以用来解决所提出的地质任务的波;干扰波:所有妨碍辨认、追踪有效波的其他波。 地震勘探中,有效波和干扰波是相对的。例如,在反射波…...
通过Wrangler CLI在worker中创建数据库和表
官方使用文档:Getting started Cloudflare D1 docs 创建数据库 在命令行中执行完成之后,会在本地和远程创建数据库: npx wranglerlatest d1 create prod-d1-tutorial 在cf中就可以看到数据库: 现在,您的Cloudfla…...
Day131 | 灵神 | 回溯算法 | 子集型 子集
Day131 | 灵神 | 回溯算法 | 子集型 子集 78.子集 78. 子集 - 力扣(LeetCode) 思路: 笔者写过很多次这道题了,不想写题解了,大家看灵神讲解吧 回溯算法套路①子集型回溯【基础算法精讲 14】_哔哩哔哩_bilibili 完…...
Objective-C常用命名规范总结
【OC】常用命名规范总结 文章目录 【OC】常用命名规范总结1.类名(Class Name)2.协议名(Protocol Name)3.方法名(Method Name)4.属性名(Property Name)5.局部变量/实例变量(Local / Instance Variables&…...
vue3 字体颜色设置的多种方式
在Vue 3中设置字体颜色可以通过多种方式实现,这取决于你是想在组件内部直接设置,还是在CSS/SCSS/LESS等样式文件中定义。以下是几种常见的方法: 1. 内联样式 你可以直接在模板中使用style绑定来设置字体颜色。 <template><div :s…...
解决本地部署 SmolVLM2 大语言模型运行 flash-attn 报错
出现的问题 安装 flash-attn 会一直卡在 build 那一步或者运行报错 解决办法 是因为你安装的 flash-attn 版本没有对应上,所以报错,到 https://github.com/Dao-AILab/flash-attention/releases 下载对应版本,cu、torch、cp 的版本一定要对…...
蓝桥杯3498 01串的熵
问题描述 对于一个长度为 23333333的 01 串, 如果其信息熵为 11625907.5798, 且 0 出现次数比 1 少, 那么这个 01 串中 0 出现了多少次? #include<iostream> #include<cmath> using namespace std;int n 23333333;int main() {//枚举 0 出现的次数//因…...
让回归模型不再被异常值“带跑偏“,MSE和Cauchy损失函数在噪声数据环境下的实战对比
在机器学习的回归分析中,损失函数的选择对模型性能具有决定性影响。均方误差(MSE)作为经典的损失函数,在处理干净数据时表现优异,但在面对包含异常值的噪声数据时,其对大误差的二次惩罚机制往往导致模型参数…...
论文笔记——相干体技术在裂缝预测中的应用研究
目录 相关地震知识补充地震数据的认识地震几何属性 相干体算法定义基本原理第一代相干体技术:基于互相关的相干体技术(Correlation)第二代相干体技术:基于相似的相干体技术(Semblance)基于多道相似的相干体…...
