天天看点

《编写高质量Python代码的59个有效方法》——第13条:合理利用try/except/else/f?inally结构中的每个代码块

本节书摘来自华章社区《编写高质量python代码的59个有效方法》一书中的第13条:合理利用try/except/else/f?inally结构中的每个代码块,作者[美]布雷特·斯拉特金(brett slatkin),更多章节内容可以访问云栖社区“华章社区”公众号查看

第13条:合理利用try/except/else/f?inally结构中的每个代码块

python程序的异常处理可能要考虑四种不同的时机。这些时机可以用try、except、else和f?inally块来表述。复合语句中的每个块都有特定的用途,它们可以构成很多种有用的组合方式(参见本书第51条)。

1.?f?inally块

如果既要将异常向上传播,又要在异常发生时执行清理工作,那就可以使用try/f?inally结构。这种结构有一项常见的用途,就是确保程序能够可靠地关闭文件句柄(还有另外一种写法,参见本书第43条)。

在上面这段代码中,read方法所抛出的异常会向上传播给调用方,而f?inally块中的handle.close方法则一定能够执行。open方法必须放在try块外面,因为如果打开文件时发生异常(例如,由于找不到该文件而抛出ioerror),那么程序应该跳过f?inally块。

2.else块

try/except/else结构可以清晰地描述出哪些异常会由自己的代码来处理、哪些异常会传播到上一级。如果try块没有发生异常,那么就执行else块。有了这种else块,我们可以尽量缩减try块内的代码量,使其更加易读。例如,要从字符串中加载json字典数据,然后返回字典里某个键所对应的值。

如果数据不是有效的json格式,那么用json.loads解码时,会产生valueerror。这个异常会由except块来捕获并处理。如果能够解码,那么else块里的查找语句就会执行,它会根据键来查出相关的值。查询时若有异常,则该异常会向上传播,因为查询语句并不在刚才那个try块的范围内。这种else子句,会把try/except后面的内容和except块本身区分开,使异常的传播行为变得更加清晰。

3.混合使用

如果要在复合语句中把上面几种机制都用到,那就编写完整的try/except/else/f?inally结构。例如,要从文件中读取某项事务的描述信息,处理该事务,然后就地更新该文件。为了实现此功能,我们可以用try块来读取文件并处理其内容,用except块来应对try块中可能发生的相关异常,用else块实时地更新文件并把更新中可能出现的异常回报给上级代码,然后用f?inally块来清理文件句柄。

这种写法很有用,因为这四块代码互相配合得非常到位。例如,即使else块在写入result数据时发生异常,f?inally块中关闭文件句柄的那行代码,也依然能执行。

要点

无论try块是否发生异常,都可利用try/f?inally复合语句中的f?inally块来执行清理工作。

else块可以用来缩减try块中的代码量,并把没有发生异常时所要执行的语句与try/except代码块隔开。

顺利运行try块后,若想使某些操作能在f?inally块的清理代码之前执行,则可将这些操作写到else块中。