第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块中。