python使用@contextmanager来定义上下文管理器

python 上下文管理器

上下文管理器是在Python2.5之后加入的功能,可以在方便的需要的时候比较精确地分配和释放资源, with便是上下文管理器的最广泛的应用; 比使用try:…finally:f.close方便的多.

with open("test/test.txt","w") as f:
	  f.write("hello")

上下文管理协议:实现了__enter__和__exit__方法。(一如迭代器协议是实现__iter__方法)
例:

class Contextor:
    def __enter__(self):
        print(1111)
        
    def __exit__(self, exc_type, exc_val, exc_tb):
        print(333)
        
	def test():
		print(222)

比如:

我们正常不使用用上下文的代码:

c = Contextor()
c.test()
# 结果如下:
>>> 222

当我们使用上下文的时候

with Contextor() as c:
	c.test()

# 结果如下
>>> 1111
>>> 222
>>> 333

类中有两个特殊的魔术方法:
enter: with语句中的代码块执行前, 会执行__enter__, 返回的值将赋值给with句中as后的变量.
exit: with语句中的代码块执行结束或出错, 会执行_exit__

如果我们要定义上下文管理器,就需要在这类里面定义__enter__和__exit__这两个方法。

有没有什么简化的方法不需要我们直接定义__enter__和__exit__呢?

python提供了一个装饰器叫@contextmanager,是位于contextlib模块下的。
借助contextmanager装饰器,我们可以简化MyResource的定义

现在我们不要__enter__和__exit__方法,只保留核心方法query(),注意我们只是不想在定义__enter__和__exit__这两个方法,但是他们里面所执行的语句我们还是需要实现的。在进入上下文管理器的时候打印__enter__里面的方法,在退出的时候打印__exit__里面的方法。

from contextlib import contextmanager


class Contextor:
    def test(self):
        print(222)

    @contextmanager
    def with_test(self):
        print(1111)
        yield self
        print(333)


with Contextor().with_test() as c:
    c.test()

# 结果如下
>>> 1111
>>> 222
>>> 333

被装饰器装饰的函数分为三部分:
with语句中的代码块执行前执行函数中yield之前代码
yield返回的内容复制给as之后的变量
with代码块执行完毕后执行函数中yield之后的代码


参考博文:
https://blog.csdn.net/qq_25473157/article/details/87103840
https://blog.csdn.net/JENREY/article/details/86690018

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值