优秀的判断力来自经验,但经验来自于错误的判断。 - 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]
KeyboardInterrupt Ctrl+C被按下
SyntaxError Python代码非法,代码不能编译
ValueError 传入一个调用者不期望的值,即使值的类型是正确的
异常处理
为了保证程序的健壮性与容错性,即在遇到错误时程序不会崩溃,我们需要对异常进行处理
如果错误发生的条件是可预知的,我们需要用if进行处理:在错误发生之前进行预防
AGE = 10
while True:
age = input('>>: ').strip()
if age.isdigit(): # 只有在age为字符串形式的整数时,下列代码才不会出错,该条件是可预知的
age = int(age)
if age == AGE:
print('you got it')
break
如果错误发生的条件是不可预知的,则需要用到try...except:在错误发生之后进行处理
异常类只能用来处理指定的异常情况,如果非指定异常则无法处理
try:
int('hello')
except IndexError as e: # 未捕获到异常,程序直接报错
print(str(e))
多分支
try:
int('hello')
except IndexError as e:
print('from IndexError: %s' % str(e))
except KeyError as e:
print('from KeyError: %s' % str(e))
except ValueError as e:
print('from ValueError: %s' % str(e))
except Exception as e:
print('from Exception: %s' % str(e))
异常的完整结构
try:
int('hello')
except Exception as e:
print('from Exception: %s' % str(e))
else:
print('try内代码块没有异常,则执行else')
finally:
print('无论异常与否,都会执行该模块,进行清理工作')
主动触发异常
try:
raise TypeError('类型错误')
except Exception as e:
print('from Exception: %s' % str(e))
自定义异常
class CustomException(BaseException):
def __init__(self, msg):
self.msg = msg
def __str__(self):
return '<CustomException: %s>' % self.msg
try:
raise CustomException('权限错误')
except CustomException as e:
print(e)
断言
两部分合作开发,确保上游满足一定条件
assert isinstance(custom, Custom)
异常处理机制
在系统内部,解释器使用一种被称作"块栈"(block stack)的结构来处理异常逻辑。在运行期提前将跳转存储到块栈,遇到异常时解释器会检查当前块栈内是否有匹配的处理逻辑,如果有则跳转并执行相应的指令;如果没有则沿调用栈向外传递,知道捕获或程序崩溃。
异常对象被保存到当前线程状态里,可用sys.exc_info查看
import sys
print(sys.exc_info()) # (None, None, None)
try:
raise Exception('err')
except:
# (<class 'Exception'>, Exception('err',), <traceback object at 0x0000027AAACFC4C8>)
print(sys.exc_info())
# 异常一旦被捕获处理,保存在线程内的exc_type、exc_value、exc_traceback都会被清除
print(sys.exc_info()) # (None, None, None)
源码分析
实际项目开发中,会根据内置异常类自定义各种功能需求类,如class RequestExeption(IOError):pass
相当于打了不同的锚点,raise异常后,就可以根据不能功能锚点做相应处理
遇到网络问题(如 DNS查询失败、拒绝连接等)时,抛出ConnectionError
HTTP请求返回不成功的状态码,r.raise_for_status()会抛出HTTPError
连接超时 ConnectTimeout,读超时ReadTimeout,基础于Timeout
请求超出最大重定向次数,抛出TooManyRedirects
requests显式抛出的异常都继承自 RequestException
项目实践
import traceback
import sys
'''
1. 复杂逻辑处理时,如客户详情大量字段,每个字段都有相应的权限、参数检查等操作,
可以继承Exception自定义不同的异常,最外层捕捉不同异常,实现分步操作
2. as e的e是异常实例,如果想追踪异常栈信息来进行相关操作 traceback
'''
class CustomBaseException(Exception):
def __init__(self, msg):
super(Exception, self).__init__(msg)
class CustomPermException(CustomBaseException):
def __init__(self):
super(CustomPermException, self).__init__('perm forbidden')
class CustomParamException(CustomBaseException):
def __init__(self, reason):
super(CustomParamException, self).__init__(reason)
try:
# raise CustomParamException('Params Error!')
# raise CustomPermException()
# raise Exception
raise Exception('msg')
except CustomPermException as e:
# print_exception、format_exception的快捷形式
traceback.print_exc()
tb_msg = traceback.format_exc()
print(tb_msg)
except CustomParamException as e:
# 打印、获取异常详细信息
traceback.print_exception(type(e), e, e.__traceback__)
tb_msg = traceback.format_exception(type(e), e, e.__traceback__)
print(tb_msg)
except Exception as e:
print(type(e)) # <class 'Exception'>
print(isinstance(e, Exception)) # True
print(str(e)) # msg 字符串
print(e.args) # ('msg',),当仅仅raise 类Exception时返回值为(,) --> 实际使用时返回实例就可以带参数,推荐
print(sys.exc_info()) # 异常类,异常类实例,异常追踪栈
etype, value, tb = sys.exc_info() # (<class 'Exception'>, Exception('msg',), <traceback object)