【妙用协程】 - 单元测试的setUp和tearDown

不少测试都须要在启动的时候作一些事情,而后在结束的时候再把作的事情给清理了。通常的作法是把这些动做写在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的逻辑了。测试

相关文章
相关标签/搜索