使用try...except可以处理异常
异常处理
import sys
try:
print('try...')
r = 10/0
print('result:',r)
except ZeroDivisionError as e:#
print('except:',e)
finally:
print('finally...')
print('end')
输出
try...
except: division by zero
finally...
end
如果在try语句块中出现错误,剩下try部分剩下的语句不会继续被执行
如果异常类型和except之后的名称相符,就会执行对应的except下的语句
如果有finally语句块,则执行finally语句块,至此,执行完毕
把0改成2,由于没有错误发生,except部分会不执行,但是finally如果有,则一定会被执行(可以没有finally语句)
捕获多个异常
try:
print('try...')
r = 10/int('a')
print('result:',r)
except ValueError as e:
print('ValueError:',e)
except ZeroDivisionError as e :
print('ZeroDivisionError:',e)
print('finally')
输出
try...
ValueError: invalid literal for int() with base 10: 'a'
finally
int()函数可能会抛出ValueError,所以我们用一个except捕获ValueError,用另一个except捕获ZeroDivisionError
一个try语句可能有多个except子句,分别来处理不同的异常
except (RuntimeError,TypeError,NameError):
# pass
一个except子句可以同时处理多个异常,这些异常将被放在一个括号里成为一个元组
可以在except后面加个else,当没有错误发生的时候,会自动执行else语句
for arg in sys.argv[1:]:
try:
f = open(arg, 'r')
except IOError:
print('cannot open', arg)
else:
print(arg, 'has', len(f.readlines()), 'lines')
f.close()
使用 else 子句比把所有的语句都放在 try 子句里面要好,这样可以避免一些意想不到的、而except又没有捕获的异常
异常处理并不仅仅处理那些直接发生在try子句中的异常,而且还能处理子句中调用的函数(甚至间接调用的函数)里抛出的异常
def foo(s):
return 10 / int(s)
def bar(s):
return foo(s) * 2
def main():
try:
bar('0')
except Exception as e:
print('Error:', e)
finally:
print('finally...')
函数main()调用foo(),foo()调用bar(),结果bar()出错了,这时,只要main()捕获到了,就可以处理:
也就是说,不需要在每个可能出错的地方去捕获错误,只要在合适的层次去捕获错误就可以了。这样一来,就大大减少了写try...except...finally的麻烦
Python的错误其实也是class,所有的错误类型都继承自BaseException,所以在使用except时需要注意的是,它不但捕获该类型的错误,还把其子类也“一网打尽”
try:
foo(2)
except ValueError as e:
print('ValueError')
except UnicodeError as e:
print('UnicodeError')
第二个except永远也捕获不到UnicodeError,因为UnicodeError是ValueError的子类,如果有,也被第一个except给捕获了
抛出异常
使用raise语句可以抛出一个指定的异常
try:
raise NameError('HiThere')
except NameError:
print('An exception flew by!')
raise
raise 唯一的一个参数指定了要被抛出的异常。它必须是一个异常的实例或者是异常的类(也就是 Exception 的子类)
只想知道这是否抛出了一个异常,并不想去处理它,那么一个简单的 raise 语句就可以再次把它抛出