Python手动抛出异常的核心在于
raise
关键字,以及你是否需要自定义异常类型。简单来说,
raise
就是告诉 Python 解释器:“这里有问题,程序必须停止!”。
使用
raise
抛出异常
如何使用
raise
raise
关键字抛出异常?
raise
关键字是 Python 中手动触发异常的利器。最简单的用法是直接
raise
一个已有的异常类实例,比如:
def check_age(age): if age < 0: raise ValueError("年龄不能为负数") # 直接抛出 ValueError 异常 print("年龄合法") check_age(-5) # 这会引发 ValueError: 年龄不能为负数
这段代码如果
age
小于 0,就会抛出一个
ValueError
异常,并附带一条错误信息。
立即学习“Python免费学习笔记(深入)”;
你也可以只
raise
异常类,Python 会自动创建一个该类的实例:
def process_data(data): if not isinstance(data, list): raise TypeError # 抛出 TypeError 异常,无需显式实例化 print("数据处理中...") process_data("不是列表") # 这会引发 TypeError
为什么要自定义异常?
自定义异常通常是为了更精确地表达程序中出现的特定错误情况。Python 内置的异常类(如
ValueError
、
TypeError
等)已经覆盖了很多常见情况,但有时你需要更具体的异常类型,以便更好地处理错误。
如何自定义异常类?
自定义异常类很简单,只需要继承 Python 内置的
Exception
类或其子类即可。例如:
class InsufficientFundsError(Exception): """当账户余额不足时抛出的异常""" def __init__(self, balance, amount): self.balance = balance self.amount = amount super().__init__(f"账户余额不足,当前余额:{balance},尝试支出:{amount}") def withdraw(balance, amount): if balance < amount: raise InsufficientFundsError(balance, amount) print("取款成功") try: withdraw(100, 200) except InsufficientFundsError as e: print(e) # 输出:账户余额不足,当前余额:100,尝试支出:200
这里,我们定义了一个
InsufficientFundsError
异常类,用于表示账户余额不足的情况。它继承自
Exception
,并添加了自己的初始化方法,可以接收余额和尝试支出的金额作为参数,并在错误信息中显示这些信息。
异常抛出后如何处理?
异常抛出后,如果没有被
try...except
块捕获,程序就会终止。因此,合理的异常处理非常重要。
def divide(a, b): try: result = a / b return result except ZeroDivisionError: print("除数不能为零") return None print(divide(10, 2)) # 输出:5.0 print(divide(10, 0)) # 输出:除数不能为零 None
在这个例子中,我们使用
try...except
块捕获了
ZeroDivisionError
异常,并进行了处理,避免了程序崩溃。
什么时候应该抛出异常?
抛出异常应该在程序遇到无法正常处理的错误时。例如:
- 输入数据不合法
- 资源无法获取(如文件不存在、网络连接失败)
- 程序逻辑出现错误
不要滥用异常,避免将异常作为正常的控制流使用。
assert
assert
和
raise
的区别是什么?
assert
语句用于在代码中插入调试断言,如果断言条件为假,则会抛出
AssertionError
异常。
assert
主要用于开发和测试阶段,用于检查代码中的假设是否成立。而
raise
则用于在运行时显式地抛出异常,表示程序遇到了无法正常处理的错误。
简单来说,
assert
是用来调试的,
raise
是用来处理错误的。
能不能在
finally
finally
块中抛出异常?
理论上可以在
finally
块中抛出异常,但这通常不是一个好主意。因为
finally
块中的异常可能会覆盖之前
try
块中抛出的异常,导致错误信息丢失。
def example(): try: print("try 块") raise ValueError("try 块异常") finally: print("finally 块") raise TypeError("finally 块异常") try: example() except TypeError as e: print(f"捕获到异常:{e}") # 输出:捕获到异常:finally 块异常
在这个例子中,
try
块抛出的
ValueError
异常被
finally
块抛出的
TypeError
异常覆盖了。因此,应该尽量避免在
finally
块中抛出异常,除非你明确知道自己在做什么。