本文主要介绍Python中的错误和异常,涉及到简单的异常处理、抛出异常以及清理动作。至于自定义异常类,将在介绍类与继承的时候讲到。
一、定义
常见的两种错误:语法错误 和 异常。
1、语法错误(Syntax Errors)
语法错误,也就是解析时错误。当我们写出不符合python语法的代码时,在解析时会报SyntaxError,并且会显示出错的那一行,并用小箭头指明最早探测到错误的位置。比如:
x = input('please input an integer:')
if int(x) > 5:
print 'hello world'
在python 3中会报语法错误:
File "/home/songlee/test", line 3
print 'hello world'
^
SyntaxError: invalid syntax
2、异常(Exceptions)
即使语句或表达式在语法上是正确的,但在尝试运行时也可能发生错误,运行时错误就叫做 异常(Exceptions) 。异常并不是致命的问题,因为我们可以在程序中对异常进行处理。
>>> 10 * (1/0)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ZeroDivisionError: division by zero
>>> 2 + x*3
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'x' is not defined
>>> '2' + 2
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: Can't convert 'int' object to str implicitly
上面展示了三种exception的类型:ZeroDivisionError、NameError、TypeError ,它们都是内置异常的名称。标准异常的名字是内建的标识符 (但并不是关键字)。
二、处理异常(try…except…)
我们可以使用 try…except… 语句来处理异常。try 语句块中是要执行的语句,except 语句块中是异常处理语句。一个 try 语句可以有多条的 except 语句,用以指定不同的异常,但至多只有一个会被执行:
try:
x = int(input('please input an integer:'))
if 30/x > 5:
print('Hello World!')
except ValueError:
print('That was no valid number. Try again...')
except ZeroDivisionError:
print('The divisor can not be zero, Try again...')
except:
print('Handling other exceptions...')
上面这段代码,当输入a(非数字)时,将抛出ValueError异常;当输入0时,将抛出ZeroDivisionError异常;当抛出其他类型的异常时,将执行except:
后的处理语句。
如果在 try 语句执行时,出现了一个异常,该语句的剩下部分将被跳过。并且如果该异常的类型匹配到了 except 后面的异常名,那么该 except 后的语句将被执行。注意,如果 except 后面没有跟异常名,表示它匹配任何类型的异常,except:
必须放在最后。
一个 except 语句可以同时包括多个异常名,但需要用括号括起来,比如:
except (RuntimeError, TypeError, NameError):
pass
try...except...语句可以有一个可选的 else 语句。else 语句必须要放在所有 except 语句后面,当
没有异常发生的时候,else 从句将被执行:
try:
name = input('please input an integer:')
f = open(name, 'r')
except IOError:
print('Cannot open', name)
except:
print('Unexpected errors.')
else:
print('close the file', name)
f.close()
三、抛出异常(raise)
raise 语句允许程序员强制地抛出一个特定的异常,例如:
>>> raise NameError('HiThere') # 抛出异常
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: HiThere
raise 抛出的异常必须是一个异常实例或类(派生自 Exception 的类)。
四、清理动作(finally)
try 语句有另一种可选的finally
从句,用于自定义一些扫尾清理的工作。
try:
x = int(input('please input an integer:'))
if x > 5:
print('Hello World!')
except ValueError:
print('It was not a number. Try again.')
finally:
print('Some clean-up actions!')
与 else 从句的区别在于: else 语句只在没有异常发生的情况下执行,而 finally 语句则不管异常发生与否都会执行。准确的说,finally 语句总是在退出 try 语句前被执行,无论是正常退出、异常退出,还是通过break、continue、return退出。
>>> def divide(x, y):
... try:
... result = x / y
... except ZeroDivisionError:
... print('error: division by zero!')
... else:
... print('executing else-clause,', 'result is', result)
... finally:
... print('executing finally-clause')
...
>>> divide(2, 1) # 正常退出
executing else-clause, result is 2.0
executing finally-clause
>>> divide(2, 0) # 异常退出
error: division by zero!
executing finally-clause
>>> divide('2', '1') # 异常退出,异常未被处理。
executing finally-clause
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 3, in divide
TypeError: unsupported operand type(s) for /: 'str' and 'str'
从上面看出,finally 语句在任何情况下都被执行了。对于没有被 except 处理的异常,将在执行完 finally 后被重新抛出。
另外,有些对象预定义了标准的清理动作(clean-up actions)。当对象不再需要时,该动作将被执行,无论对其使用的操作是否成功。例如下面的文件I/O例子:
for line in open("myfile.txt"):
print(line, end="")
这段代码的问题在于,在此代码成功执行后,文件依然被打开着。但
with
语句可以让文件对象在使用后被正常的清理掉:
with open("myfile.txt") as f:
for line in f:
print(line, end="")
在执行该语句后,文件 f 就会被关闭,就算是在读取时碰到了问题,文件 f 也会被关闭。像文件这样的对象,总会提供预定义的清理工作。