Python单元测试和Mock测试

单元测试

测试可以保证你的代码在一系列给定条件下正常工作

测试允许人们确保对代码的改动不会破坏现有的功能

测试迫使人们在不寻常条件的情况下思考代码,这可能会揭示出逻辑错误

良好的测试要求模块化,解耦代码,这是一个良好的系统设计的标志

示例

#!/usr/bin/env python

# -*- coding: utf-8 -*-

import os, sys

import time, datetime

import unittest

from unittest import TestCase

class TestSequenece(TestCase):

    def setUp(self):

        self.lst = range(10)

        print "setUp starting ..."

    def test_eq(self):

        print "test_eq starting..."

        self.assertEqual(self.lst, range(10))

    def test_in(self):

        print "test_in starting..."

        self.assertIn(1, self.lst)

        self.assertNotIn(10, self.lst)

    def test_instance(self):

        print "test_instance starting..."

        self.assertIsInstance(self.lst, list)

    def tearDown(self):

        print "tearDown starting..."

if __name__ == '__main__':

    unittest.main()

然后我们看一下执行结果再分析:

setUp starting ...

test_eq starting...

tearDown starting...

.setUp starting ...

test_in starting...

tearDown starting...

.setUp starting ...

test_instance starting...

tearDown starting...

.

----------------------------------------------------------------------

Ran 3 tests in 0.000s

OK

共运行三个测试, 每次测试成功通过都会输出一个.号

TestCase直译就是测试用例, 一个测试用例可以包含多个测试

test_xxxx就是测试项, 根据实际的功能代码逻辑来编写对应的测试项, 运行时会自动查找所有以test开发的成员函数

assertXXXX断言语句, 用来判断测试结果是否符合测试预期结果.

setUp是执行每个测试项前的准备工作, 比如:可以做一些初始化工作

tearDown是执行在每个测试项后的收尾工作,销毁测试过程中产生的垃圾, 恢复现场等

Mock测试

Mock测试是什么鬼? 我们常常遇到这样一种场景, 我们测试一些函数, 而这些函数内部调用另外带有副作用的操作, 这可能导致我们在测试过程中对数据造成未知的副作用, 而这并不是我们希望在测试中看到的.

Mock测试可以替换到指定的Python对象或者方法, 并自定义指定对象或者方法的返回值, 从来模拟对象或者方法, 消除副作用.

Mock在Python3.3时加入到标准库中, 2.X版本可以通过pip安装

$ pip install mock

首先任意写一个函数

# -*- coding: utf-8 -*-

#!/usr/bin/env python

import os, sys, time

def foo():

    lst = [1]

    lst = give_me_five(lst)

    return lst

def give_me_five(lst):

    return lst * 5

我们希望通过单元测试来测试这个函数的逻辑正确性.

# -*- coding: utf-8 -*-

#!/usr/bin/env python

import os, sys, time

#  sys.path.insert(0, os.path.dirname(os.path.abspath(__file__)))

import unittest

from unittest import TestCase

import mock

import module

class Foo(object):

    pass

class TestMock(TestCase):

    # 1

    def test_method(self):

        obj = Foo()

        obj.method = mock.MagicMock(return_value=3)

        print obj.method

        self.assertEqual(obj.method(4), 3)

    # 2

    @mock.patch('module.foo')

    def test_decorator(self, foo):

        #  res = module.foo()

        foo.return_value = [1, 2, 3]

        self.assertEqual(foo(), [1, 2, 3])

    # 3

    def test_with(self):

        with mock.patch('module.give_me_five') as give_me_five:

            give_me_five.return_value = "I'm Mock"

            self.assertEqual(module.foo(), "I'm Mock")

    # 4

    def test_module(self):

        module.give_me_five = mock.Mock(return_value=[1] * 5)

        module.give_me_five([1])  # 此时已经变成了一个Mock对象, 并尝试调用

        module.give_me_five.assert_called_with([1])  # 对mock的参数进行断言

        self.assertEqual(module.foo(), [1] * 5)

if __name__ == '__main__':

    unittest.main()

我们首先集成TestCase创建了一个单元测试

# 1位置, 我们通过mock提供的函数给obj的method方法设置返回值(可以看到类中并不包含method方法). 最后通过断言来判断返回值等于我们通过MagicMock设置的返回值

# 2位置, 我们通过mock提供的装饰器, patch()可以作为函数做装饰, 类装饰器, 上下文管理器 将module中的foo函数给mock掉, 并且并mock的函数生成的Mock对象作为类成员函数参数传入, 指定了foo函数的返回值, 并通过了断言测试

# 3位置, 将patch()作为一个上下文管理, 关于上下文管理器可以看我另一篇文章Python奇技淫巧, 用法和作为装饰器基本类似

# 4位置, 我们调用module.foo函数, 而我们并不关系foo()调用了那些函数, 我只关心在成功调用module.give_me_five后, foo函数的逻辑正确性. 所以此次我们通过Mock函数给module.give_me_five指定我们希望的返回值. 这样就能独立的测试module.foo的逻辑

mock的主要思想: 通过mock对象对某些函数进行替换, 对在测试上下文中, 这些被mock的函数被重定向到指定的mock对象

mock还有一些更高级的应用

MagicMock是Mock的子类, 并且包含一些如__str__一样的黑魔法函数, 使用MagicMock甚至可以mock掉黑魔法函数

通过patch.object可以mock掉类中指定的成员函数

通过patch.dict可以将对象mock为字典

通过patch中的start和stop方法可以控制mock的生效范围, 更加灵活的运行mock测试

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容