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

Python 异常处理完全指南

目录

    • 一、异常处理基础
      • 1. 基本语法结构
    • 二、常见异常类型
      • 1. 内置异常层次
      • 2. 常见异常示例
    • 三、多重异常处理
      • 1. 合并捕获
      • 2. 分层处理
    • 四、finally与else子句
      • 1. finally 应用场景
      • 2. else 使用技巧
    • 五、自定义异常
      • 1. 创建异常类
      • 2. 异常继承体系
    • 六、异常链与上下文
      • 1. raise from 语法
      • 2. 异常上下文查看
    • 七、最佳实践与陷阱
    • 综合实战案例


异常处理流程图

一、异常处理基础

1. 基本语法结构

try:
    # 可能引发异常的代码
    result = 10 / 0
except ZeroDivisionError:
    # 处理特定异常
    print("不能除以零!")
except Exception as e:
    # 通用异常处理
    print(f"发生错误: {e}")
else:
    # 无异常时执行
    print("计算成功")
finally:
    # 始终执行的清理代码
    print("清理资源")
  1. 异常传播机制
触发异常
当前代码有try块?
匹配except
向调用栈上层传播
执行对应处理
到达顶层?
程序崩溃

二、常见异常类型

1. 内置异常层次

所有异常类都继承 BaseException

BaseException
 ├── KeyboardInterrupt
 ├── SystemExit
 └── Exception
      ├── ArithmeticError
      │    ├── ZeroDivisionError
      │    └── FloatingPointError
      ├── LookupError
      │    ├── IndexError
      │    └── KeyError
      ├── OSError
      │    ├── FileNotFoundError
      │    └── PermissionError
      └── ValueError

2. 常见异常示例

# 文件操作
try:
    with open("nonexist.txt") as f:
        content = f.read()
except FileNotFoundError:
    print("文件不存在")

# 类型转换
try:
    num = int("abc")
except ValueError:
    print("无效的数字格式")

# 字典操作
d = {"key": "value"}
try:
    print(d["missing"])
except KeyError:
    print("键不存在")

三、多重异常处理

1. 合并捕获

try:
    # 可能抛出多种异常的代码
except (TypeError, ValueError) as e:
    print(f"输入类型错误: {e}")

2. 分层处理

try:
    data = json.loads(invalid_json)
except json.JSONDecodeError:
    print("JSON解析失败")
except Exception:
    print("其他未知错误")

四、finally与else子句

1. finally 应用场景

file = None
try:
    file = open("data.txt", "r")
    process(file)
except IOError:
    print("文件操作错误")
finally:
    if file:
        file.close()  # 确保资源释放

2. else 使用技巧


try:
    result = risky_operation()
except NetworkError:
    handle_error()
else:
    save_result(result)  # 仅在成功时执行

五、自定义异常

1. 创建异常类

class InvalidEmailError(ValueError):
    """自定义邮箱格式异常"""
    def __init__(self, email):
        super().__init__(f"无效邮箱格式: {email}")
        self.email = email

# 使用示例
email = "user@"
if "@" not in email or "." not in email.split("@")[1]:
    raise InvalidEmailError(email)

2. 异常继承体系

class DatabaseError(Exception):
    """数据库操作基类异常"""

class ConnectionError(DatabaseError):
    """数据库连接异常"""

class QueryError(DatabaseError):
    """SQL查询异常"""

六、异常链与上下文

1. raise from 语法

try:
    config = load_config()
except FileNotFoundError as e:
    raise RuntimeError("配置加载失败") from e

2. 异常上下文查看

try:
    # 可能出错的代码
except Exception as e:
    print("原始异常:", e.__cause__)
    print("上下文:", e.__context__)

七、最佳实践与陷阱

推荐做法

  1. 具体异常捕获优先
  2. 添加有意义的错误信息
raise ValueError(f"无效参数值: {value}")
  1. 日志记录异常
import logging
try:
    process()
except Exception:
    logging.exception("处理失败")

⚠️ 避免陷阱

  1. 捕获过于宽泛的异常
try:
    ...
except:  # 捕获所有异常,包括KeyboardInterrupt
    pass
  1. 忽略异常
try:
    ...
except Exception:
    pass  # 静默失败
  1. 重复抛出异常
try:
    ...
except Exception as e:
    print(e)
    raise  # 保留原始堆栈信息

综合实战案例

网络请求重试机制

import requests
from time import sleep

def fetch_data(url, retries=3):
    for attempt in range(retries):
        try:
            response = requests.get(url, timeout=5)
            response.raise_for_status()
            return response.json()
        except requests.HTTPError as e:
            print(f"HTTP错误: {e}")
        except requests.Timeout:
            print("请求超时")
        except requests.RequestException:
            print("网络连接问题")
        
        if attempt < retries - 1:
            sleep(2 ** attempt)  # 指数退避
    raise Exception(f"请求失败,已重试{retries}次")

# 使用示例
data = fetch_data("https://api.example.com/data")

通过掌握这些异常处理技巧,您可以:

  • 编写更健壮的程序
  • 提高错误诊断效率
  • 创建更友好的用户交互体验
  • 构建可维护的异常处理体系

建议在实际开发中结合具体业务场景,设计合理的异常处理策略,并定期审查异常处理代码的有效性

相关文章:

  • 基于YOLOv8深度学习的PCB缺陷检测识别系统【python源码+GUI界面+数据集+训练代码+登录界面】
  • [AI绘图] ComfyUI 中自定义节点插件安装方法
  • 雕琢健康生活,奏响养生乐章
  • 使用Geotools从DEM数据中读取指定位置的高程实战
  • 【图论】网络流算法入门
  • 初阶7 vector
  • Mybatis错误resultMap must match
  • [思路提供]Mysql主从复制时的网络延迟很高,如何调整MySQL复制参数
  • Conda命令合集【更新ing】
  • C语言基础要素(018):循环控制:break与continue
  • leetcode0525. 连续数组-medium
  • Linux shift 命令使用详解
  • C++中使用ShellExecute函数调用其他窗口程序时,参数设置为隐藏,后续能通过发消息给这个被调用程序显示,能显示出来窗口吗
  • 汽车加气站操作工证书报考条件是什么?
  • 【计算机操作系统】第一章操作系统引论
  • 优选算法系列(4.前缀和 _下) k
  • Deepseek API+Python 测试用例一键生成与导出 V1.0.4 (接口文档生成接口测试用例保姆级教程)
  • OkHttp 的证书设置
  • QOpenGLWidget视频画面上绘制矩形框
  • Redis简单介绍和安装
  • 国内锂矿“双雄”开局业绩PK:从巨亏中崛起,或深陷泥淖谋求多元转型
  • 特朗普宣布提名迈克·沃尔兹为下一任美国驻联合国大使
  • 426.8万人次!长三角铁路创单日客发量历史新高
  • 山东省委组织部办公室主任吴宪利已任德州市委常委、组织部部长
  • “80后”蒋美华任辽宁阜新市副市长
  • 马上评|什么才是地方文旅宣传的正确姿势