查看原文
其他

给妹子讲python-S01E23初识异常

给妹子讲python Python爱好者社区 2019-04-07

作者:酱油哥/ 清华程序猿      
微信公众号: python数据科学家
知乎专栏: 《给妹子讲python》
https://zhuanlan.zhihu.com/c_147297848



前文传送门:

给妹子讲python-S01E01好用的列表

给妹子讲python-S01E02学会用字典

给妹子讲python-S01E03元组的使用

给妹子讲python-S01E04容器遍历和列表解析式

给妹子讲python-S01E05字符串的基本用法

给妹子讲python-S01E06字符串用法进阶

给妹子讲python-S01E07字符编码历史观:从ASCII到Unicode

给妹子讲python-S01E08理清python中的字符编码方法

给妹子讲python-S01E09文件操作小意思

给妹子讲python-S01E10动态类型与共享引用

给妹子讲python-S01E11赋值与对象拷贝

给妹子讲python-S01E12循环迭代初体验

给妹子讲python-S01E13循环迭代高级技巧

给妹子讲python-S01E14可迭代对象和迭代器

给妹子讲python-S01E15迭代环境

给妹子讲python-S01E16生成器的使用

给妹子讲python-S01E17函数的基本特征

给妹子讲python-S01E18初探函数作用域

给妹子讲python-S01E19内嵌作用域与函数闭包

给妹子讲python-S01E20函数参数的传递与修改

给妹子讲python-S01E21函数参数的匹配与解包

给妹子讲python-S01E22神奇的装饰器

【要点抢先看】

1.异常的默认处理和主动捕获
2.主动触发异常和自定义异常
3.finally终止代码块的用法

【妹子问】从字面上来看,异常是程序运行时出现的错误吧。

没错,每当在运行时检测到程序错误时,python就会引发异常。对待异常有两种方法:一是可以在程序中捕捉和响应错误;或者忽略已发生的异常。

如果是忽略已发生的异常,python默认的异常处理行为将启动:停止程序,打印出错消息。如果不想启动这种默认行为,就要写try语句来捕捉异常并从异常中恢复,当程序运行检测到错误时,python会跳到try处理器,而程序在try之后会重新继续执行。

首先来看看python自带的默认异常处理器

def fetcher(obj, index):
    return obj[index]

x = 'spam'

print(fetcher(x,3))
print(fetcher(x,9))

m
Traceback (most recent call last):
 File "E:/12homework/12homework.py", line 7, in <module>
print(fetcher(x,9))
 File "E:/12homework/12homework.py", line 2, in fetcher
return obj[index]
IndexError: string index out of range

从这个例子可以看到,我们试图对字符串末尾以后的位置做索引运算,当函数尝试执行obj[9]时,就会触发异常。Python会替序列检测到超出边界的索引运算,并通过抛出(触发)内置的IndexError异常进行报告。

在这个例子中,我们的代码没有刻意去捕捉这个异常,所以他会一直向上返回到程序顶层,并启用默认的异常处理器:就是打印标准出错信息,即异常发生时激活的程序行和函数清单。

那么,如果我们想自己去捕获异常呢?

因为在有些情况下,这并不是我们想要的。例如,服务器程序一般需要在内部发生错误时依然保持继续工作。如果你不想要默认的异常行为,就需要把调用封装在try语句内,自行捕捉异常。

def fetcher(obj, index):
    return obj[index]

x = 'spam'

try:
    fetcher(x,9)
except IndexError:
    print('got exception')

got exception

现在,当try代码块内程序执行触发异常时,python会自动跳至处理器(即except分句下面的代码块)去运行。

def fetcher(obj, index):
    return obj[index]

x = 'spam'

try:
    fetcher(x,9)
except IndexError:
    print('got exception')
print('continue...')

got exception
continue...

在这个例子中,我们在异常捕捉和处理后,程序在捕捉了整个try语句后继续执行;这就是我们之所以得到continue消息的原因。我们没有看见标准出错信息,而程序也将正常执行下去。

除了python自身会产生异常以外,我们在程序中也可以主动引发异常。想要手动触发异常,可以直接执行raise语句。用户通过raise触发的异常的捕捉方式和python程序自身引发的异常一样:

try:
    raise IndexError
except IndexError:
    print('got exception')

got exception

如果没有去捕捉到异常,用户定义的异常就会向上传递,直到顶层默认的异常处理器,并通过标准出错信息终止该程序,看看,是不是感觉很熟悉。

raise IndexError

Traceback (most recent call last):
 File "E:/12homework/12homework.py", line 1, in <module>
raise IndexError
IndexError

我们还可以自定义异常

刚才我们利用raise语句触发了python内置作用域中定义的一个内置异常。其实我们也可以自己定义一个新的异常,这里可能需要一点面向对象的知识,所以我们只需要了解即可:自定义的异常能够通过类来编写,它继承自一个内置的异常类:通常这个类的名称叫做Exception

class Bad(Exception):
    pass

def doomed():
    raise Bad()

try:
    doomed()
except Bad:
    print('got Bad')

got Bad

最后说说终止行为finally代码块

try语句可以包含finally代码块。可以定义一定会在最后执行时的收尾行为。这里的“一定“指的是无论try代码块中是否发生了异常都会执行。

try:
    raise IndexError
finally:
    print('in finally')
print('after finally')

in finally
Traceback (most recent call last):
 File "E:/12homework/12homework.py", line 2, in <module>
raise IndexError
IndexError


try:
    print('ok')
finally:
    print('in finally')
print('after finally')

ok
in finally
after finally

可以看出,上述try/finally语句组合,无论try代码块是否发生异常,程序都将会执行finally代码块中的语句。但是当有异常发生时,python会跳过去执行finally中的行为,执行完finally中的语句后,再将try中的异常传递给顶层的默认处理器,因此finally后面的语句就不会执行了。但是如果try中的代码不触发异常,则finally后面的代码块就会正常的继续执行。

我们总结一下:

在实际应用中,try/except的组合可用于捕捉异常并从中恢复,而try/finally的组合则很方便,可以确保无论try代码块内的代码是否发生了异常,终止行为都一定会运行。

一个例子是:比如无论是否出现异常,无论异常是否被捕获,都一定会确保关闭文件。

最终我们是可以把try/except/finally三者连用的,try内为主体功能代码,except用来捕获异常,而无论异常是否出现,是否被except捕获,都将执行finally内的语句。

Python爱好者社区历史文章大合集

Python爱好者社区历史文章列表(每周append更新一次)

福利:文末扫码立刻关注公众号,“Python爱好者社区”,开始学习Python课程:

关注后在公众号内回复“课程”即可获取:

小编的Python入门免费视频课程!!!

【最新免费微课】小编的Python快速上手matplotlib可视化库!!!

崔老师爬虫实战案例免费学习视频。

陈老师数据分析报告制作免费学习视频。

玩转大数据分析!Spark2.X+Python 精华实战课程免费学习视频。


    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存