Python 如何在函数中传递异常

Python中异常通过调用栈自动向上传播,若函数未使用try-except捕获,则异常会直接传递给调用者;可通过raise重新抛出已捕获异常以执行清理操作;使用raise new_exception from original_exception可创建异常链,保留原始错误信息;极少数情况下可将异常作为返回值处理,但不推荐。合理利用异常传播机制可提升错误处理清晰度。

在 Python 中,函数本身不会“传递”异常,但你可以通过几种方式处理或让异常向上传播。关键在于理解异常如何在调用栈中传播,以及你是否要捕获、重新抛出或包装它。

1. 不捕获异常(自动向上传递)

如果你的函数调用了一个会抛出异常的代码,而你没有使用 try-except 捕获它,异常会自动向调用者传递。

def divide(a, b):
    return a / b  # 如果 b 是 0,会抛出 ZeroDivisionError

def process(x, y): result = divide(x, y) print(f"Result: {result}") return result

调用

try: process(10, 0) except ZeroDivisionError: print("除数不能为零")

这里 process() 没有处理异常,所以 ZeroDivisionError 会直接传到最外层的 try-except。

2. 捕获后再抛出(保留原始异常)

有时你想在函数中做一些清理工作,然后让调用者继续处理异常。可以使用 raise 不带参数重新抛出。

def risky_operation():
    return 1 / 0

def wrapper(): try: return risky_operation() except Exception: print("发生错误,正在清理...") raise # 重新抛出当前异常

raise 单独使用会保持原始异常类型和 traceback,调用者看到的是最初的错误位置。

3. 抛出新的异常(可保留原异常作为原因)

如果你想提供更清晰的上下文,可以抛出新异常,并把原异常链接起来。

def read_config(file_path):
    try:
        with open(file_path) as f:
            return f.read()
    except FileNotFoundError as e:
        raise RuntimeError(f"无法加载配置文件 {file_path}") from e

使用 from e 可以保留原始异常信息,有助于调试。这叫做异常链(exception chaining)。

4. 异常作为返回值(不推荐常规使用)

极少数情况下,你可能想把异常当作普通对象返回,但这不是标准做法。

def safe_divide(a, b):
    try:
        return a / b, None
    except Exception as e:
        return None, e

result, error = safe_divide(10, 0) if error: print(f"出错了:{error}")

这种方式适用于需要明确区分成功与失败结果的场景,比如某些异步或回调逻辑,但通常建议还是用异常机制。

基本上就这些。Python 的异常是自动向上传递的,你只需决定是否捕获、是否处理、是否包装或重新抛出。合理使用 raisefrom 能让你的错误处理更清晰。不复杂但容易忽略细节。