异常结构 exceptions.py

优秀的判断力来自经验,但经验来自于错误的判断。 - Fred Brooks, 著有<<人月神话>>

背景理论

什么是异常

异常就是程序运行时发生错误的信号,构成:异常的追踪信息,异常类,异常值

异常结构把功能逻辑和错误处理分开了,结构更加清晰,防止程序意外崩溃

  • 语法错误:Python解释器进行语法检测,执行前必须改正

  • 逻辑错误:运行期发生的错误

# TypeError: int 类型不可迭代
for i in 3:
    pass

# ValueError: aaa 不是有效的10进制字符
num = int('aaa')

# IndexError: 超出索引范围
li = [1, 2, 3]
li[100]

# NameError: 变量name没有定义
name

# KeyError: 没有对应键
dic = {"name": "linda"}
dic['age']

# AttributeError: Foo没有属性x
class Foo: pass
Foo.x

# ZeroDivisionError: 0除错误
str1 = 1/0

异常的种类

  • AttributeError 试图访问一个对象没有的属性,比如foo.x,但是foo没有属性x

  • IOError 输入/输出异常,基本上是无法打开文件

  • ImportError 无法引入模块或包,基本上是路径问题或名称错误

  • IndentationError 语法错误,代码没有正确对齐

  • IndexError 下标索引超出序列边界,比如当x只有三个元素,却试图访问x[5]

  • KeyError 试图访问字典里不存在的键

  • KeyboardInterrupt Ctrl+C被按下

  • NameError 使用一个还未被赋予对象的变量

  • SyntaxError Python代码非法,代码不能编译

  • TypeError 传入对象类型与要求的不符合

  • ValueError 传入一个调用者不期望的值,即使值的类型是正确的

  • ....

异常处理

为了保证程序的健壮性与容错性,即在遇到错误时程序不会崩溃,我们需要对异常进行处理

  • 如果错误发生的条件是可预知的,我们需要用if进行处理:在错误发生之前进行预防

  • 如果错误发生的条件是不可预知的,则需要用到try...except:在错误发生之后进行处理

异常类只能用来处理指定的异常情况,如果非指定异常则无法处理

多分支

异常的完整结构

主动触发异常

自定义异常

断言

两部分合作开发,确保上游满足一定条件

异常处理机制

在系统内部,解释器使用一种被称作"块栈"(block stack)的结构来处理异常逻辑。在运行期提前将跳转存储到块栈,遇到异常时解释器会检查当前块栈内是否有匹配的处理逻辑,如果有则跳转并执行相应的指令;如果没有则沿调用栈向外传递,知道捕获或程序崩溃。

异常对象被保存到当前线程状态里,可用sys.exc_info查看

源码分析

实际项目开发中,会根据内置异常类自定义各种功能需求类,如class RequestExeption(IOError):pass

相当于打了不同的锚点,raise异常后,就可以根据不能功能锚点做相应处理

  • 遇到网络问题(如 DNS查询失败、拒绝连接等)时,抛出ConnectionError

  • HTTP请求返回不成功的状态码,r.raise_for_status()会抛出HTTPError

  • 连接超时 ConnectTimeout,读超时ReadTimeout,基础于Timeout

  • 请求超出最大重定向次数,抛出TooManyRedirects

  • requests显式抛出的异常都继承自 RequestException

项目实践

Last updated