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

Python自定义异常类:实际应用示例之最佳实践

在这里插入图片描述

Python自定义异常类:实际应用示例之最佳实践

前言

在软件开发中,合理处理异常是保证程序稳定性的重要环节。虽然 Python 内置了丰富的异常类型,但在处理复杂业务逻辑时,自定义异常类能够使代码更加清晰且具备可扩展性。

本文将通过几个实际的应用场景,展示如何使用 Python 自定义异常类来增强代码的可读性与维护性。

示例1:库存管理系统中的异常处理

场景描述

在一个库存管理系统中,当用户试图购买某个商品时,系统需要检查库存是否充足。如果库存不足,系统需要抛出一个特定的异常来通知调用方进行处理,而不仅仅是抛出一个通用的异常。

解决方案

在这种场景下,我们可以定义一个 InventoryShortageException 来处理库存不足的情况。

class InventoryShortageException(Exception):def __init__(self, item_id, requested, available):self.item_id = item_idself.requested = requestedself.available = availablesuper().__init__(f"商品 {item_id} 库存不足:请求数量 {requested},仅剩 {available} 件。")class InventoryService:def check_inventory(self, item_id, requested_quantity):inventory = self.get_inventory(item_id)if inventory < requested_quantity:raise InventoryShortageException(item_id, requested_quantity, inventory)return Truedef get_inventory(self, item_id):# 模拟获取库存信息return 5# 测试
try:service = InventoryService()service.check_inventory("A1001", 10)
except InventoryShortageException as e:print(f"捕获到异常:{e}")

分析

在上述代码中,当库存不足时,InventoryShortageException 被抛出,并带有商品ID、请求数量及剩余库存等详细信息。捕获到异常后,系统可以进一步执行如通知供应商补货等后续逻辑。

优势

  • 清晰明确的异常类型提高了可读性。
  • 提供了丰富的上下文信息,便于后续处理。
  • 通过自定义异常,业务逻辑与异常处理分离,增强了系统的可维护性。

示例2:用户注册系统中的数据验证异常

场景描述

在用户注册系统中,通常需要对用户提交的数据进行验证,比如检查用户名是否已存在、邮箱格式是否正确等。为此,我们可以定义多个自定义异常类来处理不同类型的数据验证错误,使得调用方可以根据具体的错误类型做出相应处理。

解决方案

通过定义多个自定义异常类,如 UsernameAlreadyExistsExceptionInvalidEmailFormatException,可以精确地处理不同的验证错误。

class UsernameAlreadyExistsException(Exception):passclass InvalidEmailFormatException(Exception):passclass UserService:def register_user(self, username, email):if self.check_username_exists(username):raise UsernameAlreadyExistsException(f"用户名 {username} 已存在")if not self.validate_email(email):raise InvalidEmailFormatException(f"邮箱 {email} 格式不正确")# 模拟用户注册逻辑print(f"用户 {username} 注册成功")def check_username_exists(self, username):# 模拟用户名已存在的情况existing_users = ["user1", "user2"]return username in existing_usersdef validate_email(self, email):# 简单的邮箱格式校验return "@" in email and "." in email# 测试
try:service = UserService()service.register_user("user1", "example.com")
except UsernameAlreadyExistsException as e:print(f"捕获到异常:{e}")
except InvalidEmailFormatException as e:print(f"捕获到异常:{e}")

分析

在用户注册时,如果用户名已存在或邮箱格式不正确,系统分别抛出 UsernameAlreadyExistsExceptionInvalidEmailFormatException。捕获到不同类型的异常后,系统可以针对性地提示用户进行修改,确保用户体验。

优势

  • 细化了异常处理,便于调用方对不同错误做出不同反应。
  • 自定义异常类可以与具体的业务逻辑密切结合,提高代码的可读性。
  • 通过多个自定义异常,系统能更灵活地处理用户输入错误,增强了用户交互体验。

示例3:使用 traceback 定位异常位置

场景描述

在用户注册系统中,当发生数据验证错误时,开发者希望能够快速定位到抛出异常的位置,以便进行调试和修复。通过使用 traceback 模块,可以在捕获异常时输出详细的堆栈跟踪信息。

解决方案

在捕获异常时,使用 traceback.print_exc() 来输出异常的堆栈跟踪信息。

import tracebackclass UsernameAlreadyExistsException(Exception):passclass InvalidEmailFormatException(Exception):passclass UserService:def register_user(self, username, email):if self.check_username_exists(username):raise UsernameAlreadyExistsException(f"用户名 {username} 已存在")if not self.validate_email(email):raise InvalidEmailFormatException(f"邮箱 {email} 格式不正确")# 模拟用户注册逻辑print(f"用户 {username} 注册成功")def check_username_exists(self, username):# 模拟用户名已存在的情况existing_users = ["user1", "user2"]return username in existing_usersdef validate_email(self, email):# 简单的邮箱格式校验return "@" in email and "." in email# 测试
try:service = UserService()service.register_user("user1", "example.com")
except (UsernameAlreadyExistsException, InvalidEmailFormatException) as e:print(f"捕获到异常:{e}")print("堆栈跟踪信息如下:")traceback.print_exc()

分析

在这个示例中,当 UsernameAlreadyExistsExceptionInvalidEmailFormatException 被抛出时,traceback.print_exc() 会输出异常的堆栈跟踪信息。这些信息包括异常发生的文件名、行号和调用栈,帮助开发者快速定位到问题代码的位置。

优势

  • 快速定位:通过堆栈跟踪信息,开发者可以迅速找到抛出异常的具体位置。
  • 详细信息:堆栈跟踪提供了丰富的上下文信息,便于分析和解决问题。
  • 调试便利:在调试过程中,能够快速获取异常的详细信息,提高了问题解决的效率。

总结

通过本文的三个示例,我们展示了自定义异常类和 traceback 模块在复杂业务场景中的实际应用:

  1. 库存管理系统中的异常处理:通过自定义异常类 InventoryShortageException,我们能够清晰地处理库存不足的情况,并提供详细的上下文信息,便于后续处理。
  2. 用户注册系统中的数据验证异常:通过定义多个自定义异常类,如 UsernameAlreadyExistsExceptionInvalidEmailFormatException,我们可以细化异常处理,针对不同的验证错误做出相应的反应,提高用户体验。
  3. 使用 traceback 定位异常位置:在捕获异常时,使用 traceback.print_exc() 输出详细的堆栈跟踪信息,帮助开发者快速定位和解决问题。

自定义异常类和 traceback 的结合使用,不仅提高了代码的可读性和可维护性,还增强了系统的错误处理能力,使得系统具备更好的可扩展性和调试效率。

后话

本次分享到此结束,

see you~~✨✨

相关文章:

Python自定义异常类:实际应用示例之最佳实践

Python自定义异常类&#xff1a;实际应用示例之最佳实践 前言 在软件开发中&#xff0c;合理处理异常是保证程序稳定性的重要环节。虽然 Python 内置了丰富的异常类型&#xff0c;但在处理复杂业务逻辑时&#xff0c;自定义异常类能够使代码更加清晰且具备可扩展性。 本文将…...

创新设计大师项骅:用卓越才华打造医疗科技新未来

项骅,这位在设计界声名鹊起的才俊,正准备在其璀璨的职业生涯中开启一个激动人心的新篇章。近日,他宣布即将进军医疗科技领域,这一决定在设计圈和医疗界引起了广泛关注。项骅计划以UX设计师的身份,致力于改善医疗服务的用户体验。谈到这个新挑战,他显得兴致勃勃:"我期待将我…...

云计算第四阶段 CLOUD2周目 01-03

国庆假期前&#xff0c;给小伙伴们更行完了云计算CLOUD第一周目的内容&#xff0c;现在为大家更行云计算CLOUD二周目内容&#xff0c;内容涉及K8S组件的添加与使用&#xff0c;K8S集群的搭建。最重要的主体还是资源文件的编写。 (*^▽^*) 环境准备&#xff1a; 主机清单 主机…...

Linux搭建Hadoop集群(详细步骤)

前言 Hadoop是一个由Apache基金会所开发的分布式系统基础架构。用户可以在不了解分布式底层细节的情况下&#xff0c;开发分布式程序。充分利用集群的威力进行高速运算和存储。 说白了就是实现一个任务可以在多个电脑上计算的过程。 一&#xff1a;准备工具 1.1 VMware 1.2L…...

MongoDB中如何实现相似度查询

在 MongoDB 中&#xff0c;进行相似度查询通常涉及文本搜索或基于特定字段的相似度计算。以下是几种常见的方法&#xff1a; 1. 使用文本索引和文本搜索 MongoDB 提供了文本索引功能&#xff0c;可以对字符串字段进行全文搜索。你可以使用 $text 操作符来执行文本搜索查询。 …...

F开头的词根词缀:ful

60.-ful &#xff08;1&#xff09;表形容词&#xff0c;“有…的” grateful a 感激的&#xff08;grate感激&#xff09; rueful a 后悔的&#xff08;rue悔恨&#xff09; willful a 任性的&#xff08;will意志…任意办事&#xff09; tactful a 圆滑的&#xff08;tact手腕…...

【python开发笔记】-- python装饰器

装饰器&#xff1a; 不修改被装饰对象的源代码&#xff0c;也不修改调用方式的前提下&#xff0c;给被装饰对象添加新的功能 原则&#xff1a;开放封闭原则 开放&#xff1a;对扩展功能&#xff08;增加功能开放&#xff09;&#xff0c;扩展功能的意思是在源代码不做任何改变…...

WEB攻防-python考点CTF与CMS-SSTI模板注入PYC反编译

知识点&#xff1a; 1、PYC&#xff08;python编译后的文件&#xff09;文件反编译&#xff1b; 2、Python-Web-SSTI&#xff1b; 3、SSTI模板注入利用分析&#xff1b; &#xff08;Server-Side Template Injection&#xff09; SSTI 就是服务器端模板注入 当前使用的一…...

Open3D实现点云数据的序列化与网络传输

转载自个人博客&#xff1a;Open3D实现点云数据的序列化与网络传输 在处理点云数据的时候&#xff0c;有时候需要实现点云数据的远程传输。当然可以利用传输文件的方法直接把点云数据序列化成数据流进行传输&#xff0c;但Open3D源码在实现RPC功能时就提供了一套序列化及传输的…...

【C++11】右值引用

前言&#xff1a; 在C11中引入的右值引用&#xff08;rvalue references&#xff09;是现代C的一个重要特性&#xff0c;它允许开发者以更高效的方式处理临时对象&#xff08;右值&#xff09;&#xff0c;避免不必要的拷贝&#xff0c;提升性能。右值引用通常与C11的**移动语义…...

CSS元素显示类型

display 属性是 CSS 中最重要的属性之一&#xff0c;主要用来控制元素的布局&#xff0c;通过 display 属性您可以设置元素是否显示以及如何显示。 根据元素类型的不同&#xff0c;每个元素都有一个默认的 display 属性值&#xff0c;例如<div>默认的 display 属性值为 …...

Flink 介绍(特性、概念、故障容错、运维部署、应用场景)

概述 特性 概念 数据流 状态 时间 savepoint 故障容错 运维部署 部署应用到任意地方 Flink能够更方便地升级、迁移、暂停、恢复应用服务 监控和控制应用服务 运行任意规模应用 应用场景 事件驱动型应用 什么是事件驱动型应用? 事件驱动型应用的优势 Flink如何…...

Python+Flask接口判断身份证省份、生日、性别、有效性验证+docker部署+Nginx代理运行

这里写目录标题 一、接口样式二、部署流程2.1 镜像打包2.1.1 准备工作2.1.2 build打包2.1.3 dokcer部署运行2.1.4 Nginx代理 三、代码及文件3.1 index.py3.2 areaCodes.json3.3 Dockerfile 一、接口样式 https://blog.henryplus.cn/idcardApi/idCard/query?idcard{idcard} 二、…...

门店收银营销活动打折特价-收银系统源码

1.功能描述 功能描述&#xff1a;连锁店总部/门店可以将商品设置第二件打折&#xff0c;如保温杯第一件10元&#xff0c;第二件5折&#xff1b; 2.适用场景 ☑新店开业、门店周年庆、节假日等特定时间促销&#xff1b; ☑会员拉新&#xff0c;设置会员专享套餐&#xff1b; …...

QTabWidget的每个tab居中显示图标和文本

使用QTabWidget&#xff0c;给每个tab添加了图标之后&#xff0c;文字和图标之间有间距&#xff0c;没有完美居中显示。 遇到此问题&#xff0c;尝试了多种办法&#xff0c;均不理想&#xff0c;最终自定义QTabBar&#xff0c;重绘tab&#xff0c;完美解决。 #include <QT…...

Ubuntu20.04如何安装Microsoft Edge浏览器?

Microsoft Edge是由微软开发的一款网页浏览器,首次发布于2015年,作为Windows 10操作系统的默认浏览器,取代了之前的Internet Explorer。 基于Chromium内核:自2019年起,Microsoft Edge转向了使用开源的Chromium内核,这使得它与Google Chrome在性能和兼容性方面有很多相似之…...

美团Java一面

美团Java一面 9.24一面&#xff0c;已经寄了 收到的第一个面试&#xff0c;表现很不好 spring bean生命周期 作用域&#xff08;忘完了&#xff09; 为什么用redis缓存 redis和数据库的缓存一致性问题 redis集群下缓存更新不一致问题 aop说一下 arraylist和linkedlist 数据库的…...

C#中ref关键字和out关键字

值传递和引用传递 值传递和引用传递是编程中涉及数据传递的两种方式。它们的主要区别在于数据是如何在函数或方法之间传递的。 值传递 值传递意味着当你把一个变量传递给一个函数时&#xff0c;实际上传递的是这个变量的值的一个拷贝。也就是说&#xff0c;函数内部对这个参数…...

贴吧软件怎么切换ip

在网络使用中&#xff0c;有时我们需要切换IP地址来满足特定的需求&#xff0c;比如需要切换贴吧软件IP以进行不同的操作。本文将介绍几种贴吧切换IP地址的方法&#xff0c;帮助用户更好地管理自己的网络身份和访问权限。 1、更换网络环境‌ 通过连接到不同的Wi-Fi网络或使用移…...

图像分割恢复方法

传统的图像分割方法主要依赖于图像的灰度值、纹理、颜色等特征&#xff0c;通过不同的算法将图像分割成多个区域。这些方法通常可以分为以下几类&#xff1a; 1.基于阈值的方法 2.基于边缘的方法 3.基于区域的方法 4.基于聚类的方法 下面详细介绍这些方法及其示例代码。 1. 基…...

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…...

云计算——弹性云计算器(ECS)

弹性云服务器&#xff1a;ECS 概述 云计算重构了ICT系统&#xff0c;云计算平台厂商推出使得厂家能够主要关注应用管理而非平台管理的云平台&#xff0c;包含如下主要概念。 ECS&#xff08;Elastic Cloud Server&#xff09;&#xff1a;即弹性云服务器&#xff0c;是云计算…...

以下是对华为 HarmonyOS NETX 5属性动画(ArkTS)文档的结构化整理,通过层级标题、表格和代码块提升可读性:

一、属性动画概述NETX 作用&#xff1a;实现组件通用属性的渐变过渡效果&#xff0c;提升用户体验。支持属性&#xff1a;width、height、backgroundColor、opacity、scale、rotate、translate等。注意事项&#xff1a; 布局类属性&#xff08;如宽高&#xff09;变化时&#…...

MFC内存泄露

1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...

【git】把本地更改提交远程新分支feature_g

创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...

让AI看见世界:MCP协议与服务器的工作原理

让AI看见世界&#xff1a;MCP协议与服务器的工作原理 MCP&#xff08;Model Context Protocol&#xff09;是一种创新的通信协议&#xff0c;旨在让大型语言模型能够安全、高效地与外部资源进行交互。在AI技术快速发展的今天&#xff0c;MCP正成为连接AI与现实世界的重要桥梁。…...

c#开发AI模型对话

AI模型 前面已经介绍了一般AI模型本地部署&#xff0c;直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型&#xff0c;但是目前国内可能使用不多&#xff0c;至少实践例子很少看见。开发训练模型就不介绍了&am…...

【JavaSE】绘图与事件入门学习笔记

-Java绘图坐标体系 坐标体系-介绍 坐标原点位于左上角&#xff0c;以像素为单位。 在Java坐标系中,第一个是x坐标,表示当前位置为水平方向&#xff0c;距离坐标原点x个像素;第二个是y坐标&#xff0c;表示当前位置为垂直方向&#xff0c;距离坐标原点y个像素。 坐标体系-像素 …...

MySQL账号权限管理指南:安全创建账户与精细授权技巧

在MySQL数据库管理中&#xff0c;合理创建用户账号并分配精确权限是保障数据安全的核心环节。直接使用root账号进行所有操作不仅危险且难以审计操作行为。今天我们来全面解析MySQL账号创建与权限分配的专业方法。 一、为何需要创建独立账号&#xff1f; 最小权限原则&#xf…...

纯 Java 项目(非 SpringBoot)集成 Mybatis-Plus 和 Mybatis-Plus-Join

纯 Java 项目&#xff08;非 SpringBoot&#xff09;集成 Mybatis-Plus 和 Mybatis-Plus-Join 1、依赖1.1、依赖版本1.2、pom.xml 2、代码2.1、SqlSession 构造器2.2、MybatisPlus代码生成器2.3、获取 config.yml 配置2.3.1、config.yml2.3.2、项目配置类 2.4、ftl 模板2.4.1、…...