异常就是运行期检测到的错误。计算机语言针对可能出现的错误定义了异常类型,某种错误引发对应的异常时,异常处理 程序将被启动,从而恢复程序的正常运行。
准标常异
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缩进错误导致的异常TabErrorTab和空格混用SystemError一般的解释器系统异常TypeError不同类型间的无效操作ValueError传入无效的参数UnicodeErrorUnicode相关的异常UnicodeDecodeErrorUnicode解码时的异常UnicodeEncodeErrorUnicode编码错误导致的异常UnicodeTranslateErrorUnicode转换错误导致的异常
异常体系内部有层次关系,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 / yprint("result is", result)except ZeroDivisionError:print("division by zero!")finally:print("executing finally clause")divide(2, 1)# result is 2.0# executing finally clausedivide(2, 0)# division by zero!# executing finally clausedivide("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 来检查用户输入是否正确,如下:
# assertassert port >= minimum, 'Unexpected port %d when minimum was %d.' % (port, minimum)# raiseif minimum < 1024:raise ValueError('Minimum port must be at least 1024, not %d.' % (minimum,))
