单元测试

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

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

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

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

范例

#!/usr/bin/env python
# -*- coding: utf-8 -*- importos, sys
importtime, datetime
importunittest
fromunittestimportTestCase classTestSequenece(TestCase): defsetUp(self):
self.lst = range(10)
print"setUp starting ..." deftest_eq(self):
print"test_eq starting..."
self.assertEqual(self.lst, range(10)) deftest_in(self):
print"test_in starting..."
self.assertIn(1, self.lst)
self.assertNotIn(10, self.lst) deftest_instance(self):
print"test_instance starting..."
self.assertIsInstance(self.lst, list) deftearDown(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 3testsin0.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 importos, sys, time deffoo():
lst = [1]
lst = give_me_five(lst)
returnlst defgive_me_five(lst):
returnlst *5

我们希望通过单元测试来测试这个函数的逻辑正确性(请注意, 这只是一个演习!).

# -*- coding: utf-8 -*-
#!/usr/bin/env python importos, sys, time
# sys.path.insert(0, os.path.dirname(os.path.abspath(__file__)))
importunittest
fromunittestimportTestCase
importmock
importmodule classFoo(object):
pass classTestMock(TestCase):
# 1
deftest_method(self):
obj = Foo()
obj.method = mock.MagicMock(return_value=3)
printobj.method
self.assertEqual(obj.method(4),3)
# 2
@mock.patch('module.foo')
deftest_decorator(self, foo):
# res = module.foo()
foo.return_value = [1,2,3]
self.assertEqual(foo(), [1,2,3])
# 3
deftest_with(self):
withmock.patch('module.give_me_five')asgive_me_five:
give_me_five.return_value = "I'm Mock"
self.assertEqual(module.foo(), "I'm Mock")
# 4
deftest_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测试


长按二维码识别关注,您的支持是我们最大的动力。

公众号:测试梦工厂

QQ一群:300897805

  

Python单元测试和Mock测试的更多相关文章

  1. 利用Python中的mock库对Python代码进行模拟测试

    这篇文章主要介绍了利用Python中的mock库对Python代码进行模拟测试,mock库自从Python3.3依赖成为了Python的内置库,本文也等于介绍了该库的用法,需要的朋友可以参考下     ...

  2. The Hacker's Guide To Python 单元测试

    The Hacker's Guide To Python 单元测试 基本方式 python中提供了非常简单的单元测试方式,利用nose包中的nosetests命令可以实现简单的批量测试. 安装nose ...

  3. 【Python + Selenium】Mock Testing 是啥?一个so上的高票答案。

    There are many kinds of testing which really made me confused. To be honest, I've never heard of som ...

  4. [译]PyUnit—Python单元测试框架(1)

    1. 原文及参考资料 原文链接:http://docs.python.org/2/library/unittest.html# 参考文档: http://pyunit.sourceforge.net/ ...

  5. python单元测试框架笔记

    目录 单元测试概述 什么是单元测试 单元测试什么进行? 单元测试由谁负责? 单元测试需要注意 单元测试覆盖类型 python 单元测试框架 unittest pytest 测试框架 单元测试概述 什么 ...

  6. Android单元测试与模拟测试详解

    测试与基本规范 为什么需要测试? 为了稳定性,能够明确的了解是否正确的完成开发. 更加易于维护,能够在修改代码后保证功能不被破坏. 集成一些工具,规范开发规范,使得代码更加稳定( 如通过 phabri ...

  7. mock测试框架Mockito

    无论是敏捷开发.持续交付,还是测试驱动开发(TDD)都把单元测试作为实现的基石.随着这些先进的编程开发模式日益深入人心,单元测试如今显得越来越重要了.在敏捷开发.持续交付中要求单元测试一定要快(不能访 ...

  8. 在Android Studio中进行单元测试和UI测试

    本篇教程翻译自Google I/O 2015中关于测试的codelab,掌握科学上网的同学请点击这里阅读:Unit and UI Testing in Android Studio.能力有限,如有翻译 ...

  9. PYTHON单元测试

    PYTHON开发入门与实战11-单元测试 1. 单元测试 本章节我们来讲讲django工程中如何实现单元测试,单元测试如何编写以及在可持续项目中单元测试的重要性. 下面是单元测试的定义: 单元测试是开 ...

随机推荐

  1. GZipStream 压缩和解压

    GZipSteam: GZip 数据格式,它使用无损压缩和解压缩文件的行业标准算法 类 GZipStream有两种模式:CompressionMode.Compress和CompressionMode ...

  2. Uncaught ReferenceError: XXX is not defined

    Uncaught ReferenceError: XXX is not defined 这个问题困扰我很久,虽然找到了解决方法,但是还不是很明白. 如下所示:是报错的代码. 如果把它改成下面的形式就可 ...

  3. MFC程序执行顺序 .

    1.创建Application object对象theApp 程序一开始生产一个(且只有一个)Application object对象theApp,也即一个CWinApp对象,这个全局对象一产生,便执 ...

  4. opensuse 13.1 安装配置从0开始

    主要目的为自己留作备份,仅作参考! 1. 输入法 托盘->输入法->配置 去掉除英语和Sunpinyin之外的输入法,配置Sunpinyin,使用双拼方案,重启fcitx. 另外需要禁用笔 ...

  5. 指针数组 null与空字符串

    指针数组常适用于指向若干字符串,这样使字符串处理更加灵活方便. 在c++中,null表示:对象为空,它是对指针而言的.而""表示:值为空,它是对字符串而言的.

  6. 现在遇到一个问题,无法判断url是属于哪一个CDN

    解决的办法是到申请域名的网站上面去调用api获取域名的cname来确定

  7. MySQL - MySQL++在c++11环境下接口设计

    安装官方提供的mysqlconnect后,可以使用mysql++库,在官方的C API上再次做一个c++面向对象封装. 这里mysql++的安装依赖于mysql-connector-c.安装参考:ht ...

  8. 浅谈fls3的一个压缩问题

    昨天,用fis3压缩优化,在命令行运行命令的时候,出现了一个报错信息. not a PNG file. 查了很多资料,没找到解决方案,终于在github上提问,有正主回答: 是由于有些图片的后缀是通过 ...

  9. Caffe.proto使用

    参考 http://blog.csdn.net/qq_16055159/article/details/45115359 书写.proto文件 作用:编写一个 proto 文件,定义我们程序中需要处理 ...

  10. SDK截图(三):压缩位图之理论准备篇

    压缩位图我们使用简单的行程长度编码的方法,详情参考<windows程序设计>15章之DIB压缩. 在BITMAPINFOHEADER结构中有两个和位图压缩有关的字段,分别是biCompre ...