首页 > 技术文章 > 🍖异常处理

songhaixing 2021-01-14 16:24 原文

一. 什么是异常

异常就是程序运行的时候发生的错误信号, 在程序出错时, 会产生一个异常, 如果没有及时处理批该异常, 程序的运行也随即终止, 如下图 :

image-20210114114505966

两种产生异常的因素

  • 语法上的错误示例
print("aaaa"

def aaa:

class Aa

if a == "q"

for i in range(4)

......

语法上的错误在程序执行之前就应该避免, 或者说就不应该有这种错误

  • 逻辑上的错误
for i in 4:  # int类型不可迭代 : TypeError

num = input(">>")
int(num)     # 如果输入字符串无法int : ValueError

aaa          # 变量名未定义 : NameError

li = [1,2,3]
printli[5]   # 超出索引 : IndexError

dic = {"name":"shawn"}
dic["age"]   # 没有这个key : KeyError

.....

逻辑上的错误又分为 : 可控制的逻辑错误, 应该尽量解决, 避免, 而对于不可控制的逻辑错误, 我们就需要使用 异常处理 : try...except...

二.异常种类

Python 中不同的异常使用不同的类型去标识, 一个异常类型标识一种异常

常用异常类型

异常描述
TypeError

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

IndexError下标索引超出序列边界,比如当x只有三个元素,却试图访问x[5]
KeyError试图访问字典里不存在的键
AssertionErrorassert(断言)语句失败
AttributeError试图访问一个对象没有的属性,比如foo.x ,但是foo没有x这个属性。
IOError输入/输出异常,基本上是无法打开文件。
ImportError无法引入模块或者包,基本上是路径问题
IndentationError语法错误,代码没有正确对齐
KerboardInterruptCtrl + C 被按下
NameError使用一个还未被赋值予对象的变量
SyntaxErrorPython代码非法,代码不能解释
UnboundLocalError

试图访问一个还未被设置的局部变量,基本上是由于另一个同名的全局变量,导致你以为正在访问它

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

Python所有标准异常

  • 常用异常也包含在内
异常名称 描述
BaseException 所有异常的基类
SystemExit 解释器请求退出
KeyboardInterrupt 用户中断执行(通常是输入^C)
Exception 常规错误的基类
StopIteration 迭代器没有更多的值
GeneratorExit 生成器(generator)发生异常来通知退出
SystemExit Python 解释器请求退出
StandardError 所有的内建标准异常的基类
ArithmeticError 所有数值计算错误的基类
FloatingPointError 浮点计算错误
OverflowError 数值运算超出最大限制
ZeroDivisionError 除(或取模)零 (所有数据类型)
AssertionError 断言语句失败
AttributeError 对象没有这个属性
EOFError 没有内建输入,到达EOF 标记
EnvironmentError 操作系统错误的基类
IOError 输入/输出操作失败
OSError 操作系统错误
WindowsError 系统调用失败
ImportError 导入模块/对象失败
KeyboardInterrupt 用户中断执行(通常是输入^C)
LookupError 无效数据查询的基类
IndexError 序列中没有没有此索引(index)
KeyError 映射中没有这个键
MemoryError 内存溢出错误(对于Python 解释器不是致命的)
NameError 未声明/初始化对象 (没有属性)
UnboundLocalError 访问未初始化的本地变量
ReferenceError 弱引用(Weak reference)试图访问已经垃圾回收了的对象
RuntimeError 一般的运行时错误
NotImplementedError 尚未实现的方法
SyntaxError Python 语法错误
IndentationError 缩进错误
TabError Tab 和空格混用
SystemError 一般的解释器系统错误
TypeError 对类型无效的操作
ValueError 传入无效的参数
UnicodeError Unicode 相关的错误
UnicodeDecodeError Unicode 解码时的错误
UnicodeEncodeError Unicode 编码时错误
UnicodeTranslateError Unicode 转换时错误
Warning 警告的基类
DeprecationWarning 关于被弃用的特征的警告
FutureWarning 关于构造将来语义会有改变的警告
OverflowWarning 旧的关于自动提升为长整型(long)的警告
PendingDeprecationWarning 关于特性将会被废弃的警告
RuntimeWarning 可疑的运行时行为(runtime behavior)的警告
SyntaxWarning 可疑的语法的警告
UserWarning 用户代码生成的警告

更多其他异常

ArithmeticError
AssertionError
AttributeError
BaseException
BufferError
BytesWarning
DeprecationWarning
EnvironmentError
EOFError
Exception
FloatingPointError
FutureWarning
GeneratorExit
ImportError
ImportWarning
IndentationError
IndexError
IOError
KeyboardInterrupt
KeyError
LookupError
MemoryError
NameError
NotImplementedError
OSError
OverflowError
PendingDeprecationWarning
ReferenceError
RuntimeError
RuntimeWarning
StandardError
StopIteration
SyntaxError
SyntaxWarning
SystemError
SystemExit
TabError
TypeError
UnboundLocalError
UnicodeDecodeError
UnicodeEncodeError
UnicodeError
UnicodeTranslateError
UnicodeWarning
UserWarning
ValueError
Warning
ZeroDivisionError

三异常处理的完整语法

print('start...')
try:
    被监测代码1
    被监测代码2
except 异常类型1 as e:
	处理异常的代码
except 异常类型2 as e:
	处理异常的代码
except 异常类型3 as e:
	处理异常的代码
except (异常类型4,异常类型5,异常类型6) as e:
	处理异常的代码
except Exception as e:
	处理异常的代码
else:
	没有异常发生时执行的代码
finally:
	无论有无异常都会执行

print('end...')

四.异常处理的用法

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

1.首先, 如果错误的条件是可预知的, 我们就需要使用 if 进行处理, 在错误之前进行预防

num = input(">>").strip()
if num.isdigit():    # 只有用户输入的是纯数字的字符才能转成int类型进行下面的操作, 这是可预知的
    num = int(num)  
    if num > 10:
        print("ok")
    else:
        print("no")
else:
    print("请输入数字")

2.当错误发生的条件不可预知, 则需要使用 try...except 进行异常之后的处理

print("--->strat")
l = [1,2,3]
try:
    print(l[5])
except  IndexError as I:
    print(I)
print("--->end")
'''輸出
--->strat
list index out of range  # 超出索引
--->end
'''

五.try...except...的用法

1.一个异常类只能处理指定的异常, 如果非指定异常则无法检测, 程序崩溃

l = [1,2,3]
try:
    print(l[5])
except  KeyError as I:
    print(I)
# 抛出异常 : IndexError: list index out of range

2.多分支检测异常

l = [1,2,3]
try:
    print(l[5])
except  KeyError as K:
    print(K)
except ValueError as V:
    print(V)
except IndexError as I:
    print(I)
 # list index out of range

3.万能异常 : Exception

l = [1,2,3]
try:
    print(l[5])
except  Exception as E:
    print(E)
# list index out of range

4.多分支异常与万能异常

  • 如果你对检测异常的代码无论出现什么异常, 都使用一种逻辑去处理它, 那么直接使用万能异常就行了
  • 如果你的需求是对不同的异常进行不同的处理, 那么就应该使用多分支异常处理
  • 当然你也可以在多分支后面再来一个 Exception, 即你设置的不同异常都没有捕获到再使用 Exception 捕获
l = [1,2,3]
try:
    print(l[5])
except  KeyError as K:
    print(f"key--->{K}")
except ValueError as V:
    print(f"Val--->{V}")
except Exception as E:
    print(f"Exc--->{E}")
# Exc--->list index out of range

5.try...else... : 没有发生异常执行

l = [1,2,3]
try:
    print(l[2])
except  KeyError as K:
    print(f"key--->{K}")
except ValueError as V:
    print(f"Val--->{V}")
except Exception as E:
    print(f"Exc--->{E}")
else:
    print("什么异常都没有")
'''
3
什么异常都没有
'''

6.try...finally : 无论有没有发生异常都触发

推荐阅读