第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个有效方法》,仅作为个人学习使用,如有侵权请告知,将及时删除,如果觉得有益,请购买原版书籍,知识需要传递和支持,谢谢。