当前位置:   article > 正文

【Python入门】常用内建模块之contextlib_from contextlib import contextmanager

from contextlib import contextmanager

在Python中,读写文件这样的资源要特别注意,必须在使用完毕后正确关闭它们。正确关闭文件资源的一个方法是使用try...finally

  1. try:
  2. f = open('/path/to/file', 'r')
  3. f.read()
  4. finally:
  5. if f:
  6. f.close()

try...finally非常繁琐。Python的with语句允许我们非常方便地使用资源,而不必担心资源没有关闭,所以上面的代码可以简化为:

  1. with open('/path/to/file', 'r') as f:
  2. f.read()

并不是只有open()函数返回的fp对象才能使用with语句。实际上,任何对象,只要正确实现了上下文管理,就可以用于with语句。

实现上下文管理是通过__enter____exit__这两个方法实现的。例如,下面的class实现了这两个方法:

  1. class Query(object):
  2. def __init__(self, name):
  3. self.name = name
  4. def __enter__(self):
  5. print('Begin')
  6. return self
  7. def __exit__(self, exc_type, exc_value, traceback):
  8. if exc_type:
  9. print('Error')
  10. else:
  11. print('End')
  12. def query(self):
  13. print('Query info about %s...' % self.name)

这样我们就可以把自己写的资源对象用于with语句:

  1. with Query('Bob') as q:
  2. q.query()

@contextmanager

编写__enter____exit__仍然很繁琐,因此Python的标准库contextlib提供了更简单的写法,上面的代码可以改写如下:

  1. from contextlib import contextmanager
  2. class Query(object):
  3. def __init__(self, name):
  4. self.name = name
  5. def query(self):
  6. print('Query info about %s...' % self.name)
  7. @contextmanager
  8. def create_query(name):
  9. print('Begin')
  10. q = Query(name)
  11. yield q
  12. print('End')

@contextmanager这个decorator接受一个generator,用yield语句把with ... as var把变量输出出去,然后,with语句就可以正常地工作了:

  1. with create_query('Bob') as q:
  2. q.query()

很多时候,我们希望在某段代码执行前后自动执行特定代码,也可以用@contextmanager实现。例如:

  1. @contextmanager
  2. def tag(name):
  3. print("<%s>" % name)
  4. yield
  5. print("</%s>" % name)
  6. with tag("h1"):
  7. print("hello")
  8. print("world")

上述代码执行结果为:

  1. <h1>
  2. hello
  3. world
  4. </h1>

代码的执行顺序是:

  1. with语句首先执行yield之前的语句,因此打印出<h1>
  2. yield调用会执行with语句内部的所有语句,因此打印出helloworld
  3. 最后执行yield之后的语句,打印出</h1>

因此,@contextmanager让我们通过编写generator来简化上下文管理。

@closing

如果一个对象没有实现上下文,我们就不能把它用于with语句。这个时候,可以用closing()来把该对象变为上下文对象。例如,用with语句使用urlopen()

  1. from contextlib import closing
  2. from urllib.request import urlopen
  3. with closing(urlopen('https://www.python.org')) as page:
  4. for line in page:
  5. print(line)

closing也是一个经过@contextmanager装饰的generator,这个generator编写起来其实非常简单:

  1. @contextmanager
  2. def closing(thing):
  3. try:
  4. yield thing
  5. finally:
  6. thing.close()

它的作用就是把任意对象变为上下文对象,并支持with语句。

@contextlib还有一些其他decorator,便于我们编写更简洁的代码。

更加详细的解答可参考:http://www.cnblogs.com/nnnkkk/p/4309275.htmlhttps://www.ibm.com/developerworks/cn/ope 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/繁依Fanyi0/article/detail/130289
推荐阅读
相关标签
  

闽ICP备14008679号