第13条:合理利用try/expect/else/finally结构中的每个代码块

核心知识点:

(1)无论try块是否发生异常,都可以使用try/finally复合语句中地finally块来执行清理工作。

(2)顺利运行try块后,若想使某些操作能在finally块地清理代码之前执行,则可将这些操作写入到else块中

 

Python程序的异常处理可能要考虑四种不同的时机。这些时机可以用try、expect、else和finally块来表述。

复合语句中的每个块都有特定的用途,它们可以构成很多种有用的组合方式。

 

1.finally块

如果既要将异常向上传播,又要在异常发生时执行清理工作,那就可以使用try/finally结构。

这种结构有一项常见的用途,就是确保程序能够可靠地关闭文件句柄。

>>> handle = open('/tmp/my_file.txt')
>>> try:
...     data = handle.read()
... finally:
...     handle.close()

上面这段代码中,read方法所抛出的异常会向上传播给调用方,而finally块中的handle.close()方法则一定能够执行。

open方法必须放在try块外面,应为如果打开文件时发生异常,那么程序应该跳过finally块。

 

2.else块

try/expect/else结构可以清晰地描述出那些异常会由自己的代码来处理,那些异常会传播到上一级。

如果try'块没有发生异常,那么就执行else块。有了这种else块,我们可以尽量缩减try块内的代码量,使其更加易懂。

例如,要从字符串中加载JSON字典数据,然后返回字典里某个键所对应地值。

>>> def load_json_key(data,key):
...     try:
...         result_dict = json.loads(data)
...     except ValueError as e:
...         raise KeyError from e
...     else:
                return result_dict[key]

如果数据不是有效地JSON格式,那么用json.loads解码时,会产生ValueError。这个异常会由except块来捕获并处理。

如果能够解码,那么else块里地查找语句就会执行,它会根据键来查出相关地值。查询时若有异常,则该异常会向上传播,因为查询语句并不在刚才那个try块地范围内。

这种else子句,会把try/except后面地内容和except块本身区分开来,使异常地传播行为变得更加清晰。

 

 

3.混合使用

如果要在复合语句中把上面几种机制都用到,那就编写完整地try/except/else/fianally结构。

例如,要从文件中读取某项事务地描述信息,处理该事务,然后就地更新该文件。

为了实现本功能,我们可以用try块来读取文件并处理其内容,用except块来应对try中可能发生地相关问题,

用else块实时地更新文件并把更新中可能出现地问题回报给上级代码,然后用finally块来清理文件句柄。

UNDEFINED = object()

def divide_json(path):
    handle = open(path,'r+')
    try:
        data = handle.read()
        op = json.loada(data)
        value = (
            op['numerator'] /
            op['denominator'])
    except ZeroDivisionError as e:
        return UNDEFINED
    else:
        op['result'] = value
        result = json.dumps(op)
        handle.seek(0)
        handle.write(result)
        return value
    finally:
        handle.close()

这种写法很有用,因为这四块代码相互配合非常到位。

 

文章摘抄于Brett Slatkin的《编写高质量Python代码的59个有效方法》,仅作为个人学习使用,如有侵权请告知,将及时删除,如果觉得有益,请购买原版书籍,知识需要传递和支持,谢谢。

posted @ 2017-12-12 00:10  明王不动心  阅读(1745)  评论(0编辑  收藏  举报