异常就是运行期检测到的错误。计算机语言针对可能出现的错误定义了异常类型,某种错误引发对应的异常时,异常处理 程序将被启动,从而恢复程序的正常运行。
准标常异
BaseException
所有异常的类基Exception
常规异常的类基StandardError
所有的内建标准异常的基类ArithmeticError
所有数值计算异常的基类FloatingPointError
浮点计算异常OverflowError
数值运算超出最大限制ZeroDivisionError
除数为零AssertionError
断言语句(assert)失败AttributeError
尝试访问未知的对象属性EOFError
没有内建输入,到达EOF标记EnvironmentError
操作系统异常的基类IOError
输入/输出操作失败OSError
操作系统产生的异常(例如打开一个不存在的文件)WindowsError
系统调用失败ImportError
导入模块失败的时候KeyboardInterrupt
用户中断执行LookupError
无效数据查询的基类IndexError
索引超出序列的范围KeyError
字典中查找一个不存在的关键字MemoryError
内存溢出(可通过删除对象释放内存)NameError
尝试访问一个不存在的变量UnboundLocalError
访问未初始化的本地变量ReferenceError
弱引用试图访问已经垃圾回收了的对象RuntimeError
一般的运行时异常NotImplementedError
尚未实现的方法SyntaxError
语法错误导致的异常IndentationError
缩进错误导致的异常TabError
Tab和空格混用SystemError
一般的解释器系统异常TypeError
不同类型间的无效操作ValueError
传入无效的参数UnicodeError
Unicode相关的异常UnicodeDecodeError
Unicode解码时的异常UnicodeEncodeError
Unicode编码错误导致的异常UnicodeTranslateError
Unicode转换错误导致的异常
异常体系内部有层次关系,Python异常体系中的部分关系如下所示:
准标告警
Warning
警告的基类DeprecationWarning
关于被弃用的特征的警告FutureWarning
关于构造将来语义会有改变的警告UserWarning
用户代码生成的警告PendingDeprecationWarning
关于特性将会被废弃的警告RuntimeWarning
可疑的运行时行为(runtime behavior)的警告SyntaxWarning
可疑语法的警告ImportWarning
用于在导入模块过程中触发的警告UnicodeWarning
与Unicode相关的警告BytesWarning
与字节或字节码相关的警告
try-except
try
语句按照如下方式工作:
- 首先,执行
try
子句(在关键字try
和关键字except
之间的语句) - 如果没有异常发生,忽略
except
子句,try
子句执行后结束。 - 如果在执行
try
子句的过程中发生了异常,那么try
子句余下的部分将被忽略。 - 如果异常的类型和
except
之后的名称相符,那么对应的except
子句将被执行。 - 最后执行
try
语句之后的代码。 - 如果一个异常没有与任何的
except
匹配,那么这个异常将会传递给上层的try
中。
try:
f = open('test.txt')
print(f.read()) f.close()
except OSError:
print('打开文件出错')
# 打开文件出错
#---------------------------------------
try:
f = open('test.txt')
print(f.read())
f.close()
except OSError as error:
print('打开文件出错\n原因是: ' + str(error))
# 打开文件出错
# 原因是: [Errno 2] No such file or directory: 'test.txt'
一个 try
语句可能包含多个 except
子句,分别来处理不同的特定的异常。最多只有一个分支会被执行。使用多个 except
代码块时,必须坚持对其规范排序,要从最具针对性的异常到最通用的异常。
try:
int("abc")
s = 1 + '1'
f = open('test.txt')
print(f.read())
f.close()
except OSError as error:
print('打开文件出错\n原因是:' + str(error))
except TypeError as error:
print('类型出错\n原因是: ' + str(error))
except ValueError as error:
print('数值出错\n原因是: ' + str(error))
# 数值出错
# 原因是: invalid literal for int() with base 10: 'abc'
一个 except
子句可以同时处理多个异常,这些异常将被放在一个括号里成为一个元组。
try:
s = 1 + '1'
int("abc")
f = open('test.txt')
print(f.read())
f.close()
except (OSError, TypeError, ValueError) as error:
print('出错了!\n原因是: ' + str(error))
# 出错了!
# 原因是: unsupported operand type(s) for +: 'int' and 'str'
try-except-finally
不管 try
子句里面有没有发生异常, finally
子句都会执行。
如果一个异常在 try
子句里被抛出,而又没有任何的 except 把它截住,那么这个异常会在 finally
子句执行后被抛出。
def divide(x, y):
try:
result = x / y
print("result is", result)
except ZeroDivisionError:
print("division by zero!")
finally:
print("executing finally clause")
divide(2, 1)
# result is 2.0
# executing finally clause
divide(2, 0)
# division by zero!
# executing finally clause
divide("2", "1")
# executing finally clause
# TypeError: unsupported operand type(s) for /: 'str' and 'str'
try-except-else
如果在 try
子句执行时没有发生异常,Python 将执行 else
语句后的语句。
try:
fh = open("testfile", "w")
fh.write("这是一个测试文件,用于测试异常!!")
except IOError:
print("Error: 没有找到文件或读取文件失败")
else: print("内容写入文件成功")
fh.close()
# 内容写入文件成功
raise
Python 使用 raise
语句抛出一个指定的异常。
try:
raise NameError('HiThere')
except NameError:
print('An exception flew by!')
# An exception flew by!
assert
assert
是程序员用于保证程序的正确性,不是用于检查使用者输入参数是否合法。换言之,assert
可以简单的理解为程序员的 debug
工具,正式的代码中应该使用 raise
来检查用户输入是否正确,如下:
# assert
assert port >= minimum, 'Unexpected port %d when minimum was %d.' % (port, minimum)
# raise
if minimum < 1024:
raise ValueError('Minimum port must be at least 1024, not %d.' % (minimum,))