Python异常处理:自定义异常②

news2024/12/23 12:35:35

在这里插入图片描述

文章目录

    • 1. 什么是自定义异常?
    • 2. 为什么需要自定义异常?
    • 3. 如何定义自定义异常?
      • 3.1 基本自定义异常
      • 3.2 带详细信息的自定义异常
      • 3.3 自定义异常的继承层次
    • 4. 使用自定义异常
      • 4.1 抛出自定义异常
      • 4.2 捕获自定义异常
    • 5. 自定义异常的应用场景
      • 5.1 数据验证
      • 5.2 业务逻辑
      • 5.3 数据库操作
    • 6. 综合详细的例子
      • 6.1 示例代码
      • 6.2 示例解释
      • 6.3 执行结果
    • 7. 总结

在编程过程中,异常处理是一个非常重要的部分,能够有效地帮助程序员应对各种意外情况。除了Python内置的异常类型,开发者还可以创建自定义异常,以更灵活和细致地处理特定的错误情形。本文将详细介绍Python自定义异常的概念、定义方法、使用场景以及最佳实践,并附上一个综合详细的示例。

1. 什么是自定义异常?

自定义异常是指开发者根据特定需求,继承Python内置的Exception类或其子类,创建的自定义异常类。自定义异常可以提供更具体的错误信息,并使得异常处理逻辑更加清晰和有针对性。

2. 为什么需要自定义异常?

在实际开发中,内置异常类型有时无法准确表达业务逻辑中的错误情况。自定义异常可以:

  • 提供更具体和有意义的错误信息。
  • 将错误处理逻辑分离,保持代码的清晰性和可维护性。
  • 便于调试和排查问题。

3. 如何定义自定义异常?

定义自定义异常非常简单,只需创建一个继承自Exception类的新类即可。通常,自定义异常类会重写__init__方法,以便接受和存储更多的错误信息。

3.1 基本自定义异常

class MyCustomError(Exception):
    pass

3.2 带详细信息的自定义异常

class MyDetailedError(Exception):
    def __init__(self, message, error_code):
        super().__init__(message)
        self.error_code = error_code

3.3 自定义异常的继承层次

可以创建多个自定义异常类,形成继承层次结构,以便在不同的场景下使用。

class ApplicationError(Exception):
    """应用程序通用异常"""
    pass

class DatabaseError(ApplicationError):
    """数据库相关异常"""
    pass

class NetworkError(ApplicationError):
    """网络相关异常"""
    pass

4. 使用自定义异常

自定义异常的使用方法与内置异常相同,主要包括抛出(raise)和捕获(catch)两个方面。

4.1 抛出自定义异常

可以在程序中的特定位置抛出自定义异常,以便在错误发生时引发异常并传递错误信息。

def connect_to_database():
    raise DatabaseError("无法连接到数据库")

try:
    connect_to_database()
except DatabaseError as e:
    print(f"捕获到数据库异常:{e}")

4.2 捕获自定义异常

可以使用try-except语句捕获自定义异常,并根据需要处理错误。

try:
    connect_to_database()
except ApplicationError as e:
    print(f"捕获到应用程序异常:{e}")
except DatabaseError as e:
    print(f"捕获到数据库异常:{e}")
except NetworkError as e:
    print(f"捕获到网络异常:{e}")

5. 自定义异常的应用场景

自定义异常可以应用于各种场景,常见的包括:

5.1 数据验证

在数据验证过程中,可以使用自定义异常来捕获和处理无效数据。

class ValidationError(Exception):
    pass

def validate_age(age):
    if age < 0 or age > 150:
        raise ValidationError("年龄无效")

try:
    validate_age(-5)
except ValidationError as e:
    print(f"捕获到数据验证异常:{e}")

5.2 业务逻辑

在处理复杂的业务逻辑时,可以使用自定义异常来捕获和处理特定的业务错误。

class BusinessLogicError(Exception):
    pass

def process_order(order):
    if order['status'] != 'confirmed':
        raise BusinessLogicError("订单状态无效")

try:
    order = {'status': 'pending'}
    process_order(order)
except BusinessLogicError as e:
    print(f"捕获到业务逻辑异常:{e}")

5.3 数据库操作

在数据库操作中,可以使用自定义异常来捕获和处理数据库连接失败、查询错误等问题。

class DatabaseConnectionError(DatabaseError):
    pass

def connect_to_db():
    # 模拟数据库连接失败
    raise DatabaseConnectionError("数据库连接失败")

try:
    connect_to_db()
except DatabaseConnectionError as e:
    print(f"捕获到数据库连接异常:{e}")

6. 综合详细的例子

下面是一个综合详细的例子,展示了如何在一个简单的图书管理系统中使用自定义异常来处理各种错误情况。

6.1 示例代码

class LibraryError(Exception):
    """图书馆通用异常"""
    pass

class BookNotFoundError(LibraryError):
    """图书未找到异常"""
    def __init__(self, title):
        super().__init__(f"图书未找到:{title}")
        self.title = title

class BookAlreadyExistsError(LibraryError):
    """图书已存在异常"""
    def __init__(self, title):
        super().__init__(f"图书已存在:{title}")
        self.title = title

class InvalidBookError(LibraryError):
    """无效图书异常"""
    def __init__(self, title, reason):
        super().__init__(f"无效图书:{title},原因:{reason}")
        self.title = title
        self.reason = reason

class Book:
    def __init__(self, title, author, year):
        self.title = title
        self.author = author
        self.year = year

class Library:
    def __init__(self):
        self.books = {}

    def add_book(self, book):
        if book.title in self.books:
            raise BookAlreadyExistsError(book.title)
        if not book.title or not book.author or not book.year:
            raise InvalidBookError(book.title, "信息不完整")
        self.books[book.title] = book

    def remove_book(self, title):
        if title not in self.books:
            raise BookNotFoundError(title)
        del self.books[title]

    def get_book(self, title):
        if title not in self.books:
            raise BookNotFoundError(title)
        return self.books[title]

    def list_books(self):
        return list(self.books.values())

def log_activity(func):
    def wrapper(*args, **kwargs):
        try:
            result = func(*args, **kwargs)
            return result
        except LibraryError as e:
            print(f"图书馆操作异常:{e}")
            return None
    return wrapper

@log_activity
def main():
    library = Library()

    # 添加图书
    try:
        book1 = Book("Python编程", "Guido van Rossum", 2020)
        library.add_book(book1)
    except LibraryError as e:
        print(f"添加图书时发生异常:{e}")

    try:
        book2 = Book("", "Unknown Author", 2021)
        library.add_book(book2)
    except LibraryError as e:
        print(f"添加图书时发生异常:{e}")

    # 列出图书
    print("当前图书列表:")
    for book in library.list_books():
        print(f"标题:{book.title}, 作者:{book.author}, 出版年份:{book.year}")

    # 获取图书
    try:
        book = library.get_book("Python编程")
        print(f"获取图书:标题:{book.title}, 作者:{book.author}, 出版年份:{book.year}")
    except LibraryError as e:
        print(f"获取图书时发生异常:{e}")

    try:
        book = library.get_book("Java编程")
        print(f"获取图书:标题:{book.title}, 作者:{book.author}, 出版年份:{book.year}")
    except LibraryError as e:
        print(f"获取图书时发生异常:{e}")

    # 删除图书
    try:
        library.remove_book("Python编程")
        print("图书'Python编程'已删除")
    except LibraryError as e:
        print(f"删除图书时发生异常:{e}")

    try:
        library.remove_book("Java编程")
        print("图书'Java编程'已删除")
    except LibraryError as e:
        print(f"删除图书时发生异常:{e}")

if __name__ == "__main__":
    main()

6.2 示例解释

  1. 自定义异常类

    • LibraryError是图书馆通用异常的基类。
    • BookNotFoundErrorBookAlreadyExistsErrorInvalidBookError分别表示图书未找到、图书已存在和无效图书的异常。
    • 这些异常类通过继承LibraryError实现,方便在图书管理系统中统一处理。
  2. Book 类

    • Book类表示一本图书,包含标题、作者和出版年份等属性。
  3. Library 类

    • Library类实现了图书的添加、删除、获取和列出操作,并

在适当的地方抛出自定义异常。

  1. log_activity 装饰器

    • log_activity装饰器用于捕获和记录图书管理系统中的操作异常。
  2. main 函数

    • main函数是程序的入口,展示了如何在图书管理系统中使用自定义异常来处理各种错误情况,包括添加、删除和获取图书的操作。

6.3 执行结果

在这里插入图片描述

7. 总结

本文详细介绍了Python自定义异常的基本概念、定义方法、使用场景以及最佳实践。通过自定义异常,开发者可以提供更具体和有意义的错误信息,将错误处理逻辑分离,保持代码的清晰性和可维护性。最后,我们通过一个综合详细的例子展示了如何在实际应用中使用自定义异常来实现一个简单的图书管理系统。希望本文对您理解和应用Python的自定义异常有所帮助。


欢迎点赞|关注|收藏|评论,您的肯定是我创作的动力

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2143974.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【C++】透析string类

个人主页&#xff1a;CSDN_小八哥向前冲~ 所属专栏&#xff1a;C入门 目录 string类介绍 auto和范围for auto关键字 范围for string类常用接口说明 string类常见构造 string类容量操作 string类的访问及遍历操作 string类修改操作 string的结构说明 vs下的结构 G下的…

LibJPEG库使用_通过LibJPEG将RGB数据保存为JPG图片存储到磁盘

一、前言 LibJPEG库是一个广泛使用的开源C库&#xff0c;用于处理JPEG图像的压缩和解压缩。该库由独立JPEG小组&#xff08;Independent JPEG Group, IJG&#xff09;开发&#xff0c;提供了功能强大的API&#xff0c;用于创建和读取JPEG文件。LibJPEG库支持JPEG的所有常见功能…

Cpp快速入门语法(上)(1)

文章目录 前言一、C关键字(C98)二、命名空间命名空间的定义命名空间的使用 三、C输入 & 输出四、缺省参数总结 前言 其实有时候我也会尝试代入下祖师爷本杰明当年在贝尔实验室的心理活动&#xff0c;我心想&#xff0c;他可能一开始是大抵受不了C语言的某些缺点&#xff0c…

梦想之家|AI技术赋能家居,重塑生活空间

人工智能&#xff08;AI&#xff09;在智能家居方面的应用非常广泛&#xff0c;极大地提升了家庭的便利性、安全性和能源效率。当前&#xff0c;AI技术的迅速发展&#xff0c;为传统家居产品带来了智能化升级。从智能单品到智能互联&#xff0c;AI技术的融入使得这些家居产品具…

DC-DC降压10A电源降压可调模块24V转12V9V5V3V-AH1514芯片

AH1514&#xff1a;一款高效率小体积的DC-DC降压电源芯片 摘要&#xff1a;本文介绍了一款高性能的DC-DC降压电源芯片——AH1514&#xff0c;该芯片具有24V转12V、9V、5V、3V可调输出&#xff0c;支持7V-38V输入&#xff0c;20A峰值输出电流&#xff0c;且具有小体积、高效率的…

C++速通LeetCode简单第19题-只出现一次的数字

方法一&#xff1a;暴力求解&#xff0c;排序后两个两个比较&#xff0c;两者不同时前者为答案&#xff1a; class Solution { public:int singleNumber(vector<int>& nums) {if(nums.size() 1) return nums[0];list<int> l;int ans 0;for(int i 0;i< n…

3.js - THREE.CubeTextureLoader() 添加环境纹理,以创建立方体贴图

使用 THREE.CubeTextureLoader() 添加环境纹理&#xff0c;以创建立方体贴图 不使用 THREE.CubeTextureLoader() 的时候 源码 import * as THREE from three import { OrbitControls } from three/examples/jsm/controls/OrbitControls import { RGBELoader } from three/exam…

【话题讨论】AI时代程序员核心力:技术深耕,跨界学习,软硬兼备

目录 引言 一、AI辅助编程对程序员工作的影响 1.1 AI工具如何提升工作效率 1.2 AI工具的风险 1.3 应对策略 二、程序员应重点发展的核心能力 2.1 核心竞争力 2.2 企业和教育机构的调整 三、人机协作模式下的职业发展规划 3.1 持续学习的重要性 3.2 选择适合自己的…

电脑提示‘由于找不到 msvcr120.dll,无法继续执行代码’的科学解决方案分析

如果你在启动特定的应用程序或游戏时遇到错误提示&#xff1a;“由于找不到 msvcr120.dll&#xff0c;无法继续执行代码”&#xff0c;这表明你的系统可能缺少运行某些基于 Visual C 2013 开发的程序所需的关键组件。不过&#xff0c;不必担心&#xff0c;有几种方法可以解决这…

使用C++程序编写5 个浮点数,求平均值后输出

源代码如下&#xff1a; #include <iostream>using namespace std;int main() {float arr[5]{7,10,3,9,8};int i;float sum 0;float avg 0;for(i0;i<5;i){sum sum arr[i];}avg sum/5;cout << "平均值是&#xff1a;" << avg << endl…

VoIP协议

VoIP协议是VoIP业务的规范标准。我们都知道VoIP业务有着压倒性的优势。随着网络应用的多元化和低成本化发展&#xff0c;VoIP业务直接冲击着传统通信市场&#xff0c;那么目前VoIP协议目前常用的协议,如H.323、SIP、MEGACO和MGCP。 H.248 H.248是定义网关控制协议的ITU建议书…

squid代理及常见的代理上网(Squid Proxy and Common Proxy Internet Access)

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:Linux运维老纪的首页…

【学术会议征稿】第九届计算机技术与机械电气工程国际学术论坛(ISCME 2024)

第九届计算机技术与机械电气工程国际学术论坛&#xff08;ISCME 2024&#xff09; 2024 9th International Seminar on Computer Technology, Mechanical and Electrical Engineering 第九届计算机技术与机械电气工程国际学术论坛&#xff08;ISCME 2024&#xff09;将于2024…

还在为机器学习中,层出不穷的概念烦恼么?不妨看看这边,南瓜书第1,2章学习总结-task01

第一章&#xff1a;绪论 这里面讲了机器学习的基本概念&#xff0c;包括基本术语 1.1.基本概念 数据集和样本集的区别。原始检测数据是总体,总体是统计所研究对象的全体,是包含所研究变量的全部个体的集合,具有同质性、差异性、大量性。构成总体的个别事物叫总体单位。样本检测…

idea插件开发的第四天-完善JSON工具

介绍 Demo说明 本文基于maven项目开发,idea版本为2022.3以上,jdk为1.8本文在Tools插件之上进行开发本次demo将使用idea的一些组件优化 Tools插件说明 Tools插件是一个Idea插件,此插件提供统一Spi规范,极大的降低了idea插件的开发难度,并提供开发者模块,可以极大的为开发者开…

nodejs 009: 使用nvm进行node版本管理(包括Could not retrieve的手动处理办法)

nvm 有些问题类似&#xff1a;“v8::Object::Set": No overloaded function accepts 2 Arguments”可能需要通过更换nodejs的版本来解决&#xff08;如下图所示&#xff0c;需要看当时的项目的时间&#xff0c;查找当时的流行nodejs版本&#xff09;。这时可以使用nvm进行…

干耳朵里的耳屎结坨了怎么弄出来?可视挖耳勺推荐

干耳朵里的耳屎结坨了怎么弄出来&#xff1f;这个是很多干耳朵小伙伴的一个难题。用棉签掏根本掏不出来&#xff0c;反而会越推越进。用普通耳勺掏不仅不够精准还会因为盲掏&#xff0c;弄伤自己的耳膜或者刮破耳道。所以干耳朵里的耳屎结坨了&#xff0c;建议用可视挖耳勺来掏…

robosuite基础教程(一)——基本概念

robosuite和robomimic都是由ARISE Initiative开发的开源工具&#xff0c;旨在推进机器人学习和机器人操作领域的研究。 一、基本概念 robosuite是一个由MuJoCo物理引擎驱动的模拟框架&#xff0c;专为机器人学习设计。它提供了一套基准环境&#xff0c;是Advancing Robot Int…

C++/CLI编程知识点小记

1.前言 本篇博文并非详细的C/CLI教程&#xff0c;仅是博主就学习和实践总结的部分知识点记录。 第一次接触C/CLI是2017年了&#xff0c;用C编写底层库&#xff0c;C/CLI编写wrapper层&#xff0c;在C#项目中进行调用&#xff0c;开发应用。 2.内容 C/CLI是一种混合编程&…