上篇博文简单的讲了下with语句的用法以及上下文管理器对象的概念,想必对上下文管理器对象有一定的了解。一个对象如果实现了上下文管理协议,也就是在类中定义了__enter__()和__exit__()方法两个方法的对象,都可以称为上下文管理器对象。

    但是,Python中有个contextlib模块,是个比with优美的东西,提供上下文机制的,它是通过Generator装饰器实现的,不再是采用__enter__和__exit__。

   contextlib模块对外有三个接口,分别是contextlib.contextmanager,contextlib.nested,

contextlib.closing。

(1)contextlib.contextmanager

contextmanager装饰器,装饰的函数必须是一个生成器。然后返回一个函数,在函数调用的时候返回一个上下文管理器,是一种针对函数级别的上下文管理机制。常用方式如下:

#coding=utf-8
from contextlib import contextmanager

@contextmanager
def make_context() :
  print 'enter'
  try :
    yield {}
  except RuntimeError, err :
    print 'error' , err
  finally :
    print 'exit'

with make_context() as value :
  print value

输出结果:

enter
{}
exit

目前也没找到好的使用例子,以后有的话,会再贴上。

(2)contextlib.nested

nested为了一次调用多个上下文管理器的时候用,将所有上下文管理器里的__enter__放到vars里,一次yield返回,然后unpack。就产生多个对象了。__exit__都放到exits列表里,在finally里统一调用,这些__exit__在调用上下文管理器产生异常时,如何处理。如果返回false,将会把异常继续抛出,返回true则不会抛出异常。但是nested已经过时了,因为with已经可以通过多个上下文的直接嵌套了。用法如下例:

#coding=utf-8
from contextlib import contextmanager,nested

@contextmanager
def make_context(name) :
    print 'enter', name
    yield name
    print 'exit', name

with nested(make_context('A'), make_context('B')) as (a, b) :
    print a
    print b

with make_context('A') as a, make_context('B') as b :
    print a
    print b

输出结果:

enter A
enter B
A
B
exit B
exit A
enter A
enter B
A
B
exit B
exit A

(3)contextlib.closing

文件类支持上下文管理器, 但是有一些对象不支持。还有一些类使用close()方法但是不支持上下文管理器,使用closing()来为他创建一个上下文管理器。就是把有close方法却没有__exit__方法的对象变成一个上下文管理器。

看下面代码:

from contextlib import closing
class Door(object) :
    def open(self) :
        print 'Door is opened'
    def close(self) :
        print 'Door is closed'

with closing(Door()) as door :
    door.open()