With语句以及@contextmanager的语法解析
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了With语句以及@contextmanager的语法解析相关的知识,希望对你有一定的参考价值。
with 语句以及@contextmanager的语法解析
with语句可以通过很简单的方式来替try/finally
语句。 with语句中EXPR部分必须是一个包含__enter__()
和__exit__()
方法的对象,也就是Context Manager。使用with语句的目的:
- 提供可靠的资源自动释放,在with代码执行前请求资源,代码运行结束后资源会释放。
- 简化代码,代码可读性以及逻辑的简明都会提高很多。
- 创造临时的上下文环境,例如做一个临时的网络请求并获取返回值作为上下文环境。
- 通过contextmanager和generator创造线程操作异步所。
下述例子描述with...as语句的实现原理:
实现原理:
- 在with语句中, EXPR必须是一个包含
__enter__()
和__exit__()
方法的对象(Context Manager)。 - 调用EXPR的
__enter__()
方法申请资源并将结果赋值给VAR变量。 - 通过
try/except
确保代码块BLOCK正确调用,否则调用EXPR的__exit__()
方法退出并释放资源。 - 在代码块BLOCK正确执行后,最终执行EXPR的
__exit__()
方法释放资源。
EXPR可以使用with语句的前提,必须是一个包含__enter__()
和__exit()__
方法的对象(Context Manager),最直接的方式是声明一个对象,在__enter__()
方法里面申请资源,在__exit__()
方法里面释放资源;EXPR返回此对象。
更通用和更高效的将普通的函数转变为包含__enter__()
和__exit__()
方法的对象的方法是:通过一个特定的decorator(@contextmanager)扩展该函数并将函数声明为非循环的单一返回值的generator。
generator可以将函数变成类似于iterator,每次调用好像通过iterator的next逐步读取,而不是一次返回。
- 比实现一个iterator简单,iterator需要实现
__init__
,__iter__
,__next__
函数。- 比将结果一次返回(全部读取到内存中)要节省内存,通过next可以逐步获取需要的值。
@contextmanagerdecorator的实现原理:
声明contextmanager的decorator函数。参数是generator,返回值是一个接受和generator函数同样参数并且将generaor函数和参数传递到Context Manager构造函数并返回Context Manager对象的函数。绕死了!!!
decorator函数,是接受函数作为参数,并且返回一个函数的的函数。当对函数func进行此修饰时相当于对func进行一次转变: func = decorator(func),在这里generator被修饰后变成了
help(*args, **kwargs)
函数在
__enter__()
方法中使用generator的next()方法获取第一个返回值。如果gen并不是generator函数,抛出一个runtime异常。- 在
__exit__()
方法中,如果存在异常将异常跑出,否则继续调用generator的next()方法。因此generator是一个具有唯一值非loop的generator因此会抛出stopiteration异常(正常预期值),否则抛出一个runtime异常。