不少测试都须要在启动的时候作一些事情,而后在结束的时候再把作的事情给清理了。通常的作法是把这些动做写在setUp和tearDown的两个方法里,单元测试框架会负责在开始和结束的时候调用这两个方法。app
class SomeTest(unittest.case.TestCase): def setUp(self): super(SomeTest, self).setUp() setup_db() def tearDown(self): clean_db() super(SomeTest, self).tearDown()
这种写法有好几个烦人的地方。首先是Logic Locality很差的问题:setup_db()和clean_db()是分在两处的,中间可能隔着很长一段代码。从视觉上没法直观的指导setup_db()原来和clean_db()是一对的。
其次是很难重用的问题(上纲上线的话就是复杂度很差管理的问题),为了不重复写公共的setUp和tearDown通常会抽取出一个UsingDbTest这样的基类。这样全部的子类必须记得super(xxx, self).setUp(),不然就会覆盖掉基类的setUp。其次在须要有多个维度的东西须要复用的时候,好比有一个UsingDbTest的基类,有一个UsingNetworkTest的基类,难道让子类继承两个基类么(mixin是否是有点过于复杂了?)。
使用generator能够很好的解决这个问题。首先咱们写一个方法来作setUp和tearDown:框架
@contextlib.contextmanager def using_db(): setup_db() yield clean_db()
这样能够很是清晰地知道setup_db和clean_db是一对的。而后再把这个小的上下文附着到主测试逻辑上:单元测试
def apply_context(test, contextmanager): contextmanager.__enter__() test.addCleanup(lambda: contextmanager.__exit__(None, None, None)) class SomeTest(unittest.case.TestCase): def setUp(self): apply_context(self, using_db())
这里利用了单元测试的addCleanup的特性,把tearDown转化为回调在setUpd的时候就设置好。利用这种方式,咱们能够用组合的方式而不是继承的方式来复用公共的setUp和tearDown的逻辑了。测试