详解Python中的测试工具


Posted in Python onJune 09, 2019

当我们在写程序的时候,我们需要通过测试来验证程序是否出错或者存在问题,但是,编写大量的测试来确保程序的每个细节都没问题会显得很繁琐。在Python中,我们可以借助一些标准模块来帮助我们自动完成测试过程,比如:

  • unittest: 一个通用的测试框架;
  • doctest: 一个更简单的模块,是为检查文档而设计的,但也非常适合用来编写单元测试。

下面,笔者将会简单介绍这两个模块在测试中的应用。

doctest

doctest模块会搜索那些看起来像是python交互式会话中的代码片段,然后尝试执行并验证结果。下面我们以doctest.testmod为例,函数doctest.testmod会读取模块中的所有文档字符串,查找看起来像是从交互式解释器中摘取的示例,再检查这些示例是否反映了实际情况。

我们先创建示例代码文件test_string_lower.py,完整代码如下:

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

def string_lower(string):
  '''
  返回一个字符串的小写
  :param string: type: str
  :return: the lower of input string
  >>> string_lower('AbC')
  'abc'
  >>> string_lower('ABC')
  'abc'
  >>> string_lower('abc')
  'abc'
  '''
  return string.lower()

if __name__ == '__main__':
  import doctest, test_string_lower
  doctest.testmod(test_string_lower)

首先先对程序进行说明,函数string_lower用于返回输入字符串的小写,函数中的注释中,一共包含了3个测试实例,期望尽可能地包含各种测试情况,接着在主函数中导入doctest, test_string_lower,再运行doctest中的testmod函数即可进行测试。

接着,我们开始测试。首先,在命令行中输入 python test_string_lower.py ,运行后会发现什么都没有输出,但这其实是件好事,它表明程序中的所有测试都通过了!那么,如果我们想要获得更多的输出呢?可在运行脚本的时候增加参数 -v ,这时候命令变成 python test_string_lower.py -v ,输出的结果如下:

Trying:
  string_lower('AbC')
Expecting:
  'abc'
ok
Trying:
  string_lower('ABC')
Expecting:
  'abc'
ok
Trying:
  string_lower('abc')
Expecting:
  'abc'
ok
1 items had no tests:
  test_string_lower
1 items passed all tests:
  3 tests in test_string_lower.string_lower
3 tests in 2 items.
3 passed and 0 failed.
Test passed

可以看到,程序测试的背后还是发生了很多事。接着,我们尝试着程序出错的情况,比如我们不小心把函数的返回写成了:

return string.upper()

这其实是返回输入字符串的大写了,而我们测试的实例却返回了输入字符串的小写,再运行该脚本(加上参数 -v ),输出的结果如下:

Failed example:
  string_lower('abc')
Expected:
  'abc'
Got:
  'ABC'
1 items had no tests:
  test_string_lower
**********************************************************************
1 items had failures:
  3 of  3 in test_string_lower.string_lower
3 tests in 2 items.
0 passed and 3 failed.
***Test Failed*** 3 failures.

这时候,程序测试失败,它不仅捕捉到了bug,还清楚地指出错误出在什么地方。我们不难把这个程序修改过来。

关于doctest模块的更详细的使用说明,可以参考网址: https://docs.python.org/2/lib... 。

unittest

 unittest类似于流行的Java测试框架JUnit,它比doctest更灵活,更强大,能够帮助你以结构化的方式来编写庞大而详尽的测试集。

我们以一个简单的示例入手,首先我们编写my_math.py脚本,代码如下:

# -*- coding: utf-8 -*-
def product(x, y):
  '''
  :param x: int, float
  :param y: int, float
  :return: x * y
  '''
  return x * y

该函数实现的功能为:输入两个数x, y, 返回这两个数的乘积。接着是test_my_math.py脚本,完整的代码如下:

import unittest, my_math

class ProductTestcase(unittest.TestCase):

  def setUp(self):
    print('begin test')

  def test_integers(self):
    for x in range(-10, 10):
      for y in range(-10, 10):
        p = my_math.product(x, y)
        self.assertEqual(p, x*y, 'integer multiplication failed')

  def test_floats(self):
    for x in range(-10, 10):
      for y in range(-10, 10):
        x = x/10
        y = y/10
        p = my_math.product(x, y)
        self.assertEqual(p, x * y, 'integer multiplication failed')

if __name__ == '__main__':
  unittest.main()

函数unittest.main负责替你运行测试:在测试方法前执行setUp方法,示例化所有的TestCase子类,并运行所有名称以test打头的方法。assertEqual方法检车指定的条件(这里是相等),以判断指定的测试是成功了还是失败了。

接着,我们运行前面的测试,输出的结果如下:

begin test
.begin test
.
----------------------------------------------------------------------
Ran 2 tests in 0.001s

OK

可以看到,该程序运行了两个测试,每个测试前都会输出'begin test', . 表示测试成功,若测试失败,则返回的是 F 。

接着模拟测试出错的情形,将my_math函数中的product方法改成返回:

return x + y

再运行测试脚本,输出的结果如下:

begin test
Fbegin test
F
======================================================================
FAIL: test_floats (__main__.ProductTestcase)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "test_my_math.py", line 20, in test_floats
    self.assertEqual(p, x * y, 'integer multiplication failed')
AssertionError: -2.0 != 1.0 : integer multiplication failed

======================================================================
FAIL: test_integers (__main__.ProductTestcase)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "test_my_math.py", line 12, in test_integers
    self.assertEqual(p, x*y, 'integer multiplication failed')
AssertionError: -20 != 100 : integer multiplication failed

----------------------------------------------------------------------
Ran 2 tests in 0.001s

FAILED (failures=2)

两条测试都未通过,返回的是 F ,并帮助你指出了错误的地方,接下来,你应该能快速地修复这个bug。

关于unittest模块的更加详细的说明,可以参考网址: https://docs.python.org/3/lib... 。

总结

以上所述是小编给大家介绍的Python中的测试工具,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会技术回复大家的!

Python 相关文章推荐
Python计算已经过去多少个周末的方法
Jul 25 Python
12步入门Python中的decorator装饰器使用方法
Jun 20 Python
ansible作为python模块库使用的方法实例
Jan 17 Python
python 专题九 Mysql数据库编程基础知识
Mar 16 Python
Python 实现链表实例代码
Apr 07 Python
python实现折半查找和归并排序算法
Apr 14 Python
Python如何通过subprocess调用adb命令详解
Aug 27 Python
Python实现两个list求交集,并集,差集的方法示例
Aug 02 Python
Python3.6+selenium2.53.6自动化测试_读取excel文件的方法
Sep 06 Python
对Tensorflow中tensorboard日志的生成与显示详解
Feb 04 Python
Python实现从N个数中找到最大的K个数
Apr 02 Python
Pytorch转onnx、torchscript方式
May 25 Python
Python中函数参数匹配模型详解
Jun 09 #Python
Python程序包的构建和发布过程示例详解
Jun 09 #Python
Python面向对象之继承和多态用法分析
Jun 08 #Python
Python基本数据结构之字典类型dict用法分析
Jun 08 #Python
Python学习笔记基本数据结构之序列类型list tuple range用法分析
Jun 08 #Python
Python面向对象之类的封装操作示例
Jun 08 #Python
Python面向对象之类和实例用法分析
Jun 08 #Python
You might like
php adodb连接不同数据库
2009/03/19 PHP
PHP的魔术常量__METHOD__简介
2014/07/08 PHP
IE Firefox 使用自定义标签的区别
2009/10/15 Javascript
js中定义一个变量并判断其是否为空的方法
2014/05/13 Javascript
浅谈javascript对象模型和function对象
2014/12/26 Javascript
javascript结合Flexbox简单实现滑动拼图游戏
2016/02/18 Javascript
AngularJS Phonecat实例讲解
2016/11/21 Javascript
Bootstrap基本插件学习笔记之按钮(21)
2016/12/08 Javascript
Bootstrap3 多选和单选框(checkbox)
2016/12/29 Javascript
easyui datebox 时间限制,datebox开始时间限制结束时间,datebox截止日期比起始日期大的实现代码
2017/01/12 Javascript
Vue 2.0中生命周期与钩子函数的一些理解
2017/05/09 Javascript
vue中如何实现变量和字符串拼接
2017/06/19 Javascript
node.js中grunt和gulp的区别详解
2017/07/17 Javascript
基于daterangepicker日历插件使用参数注意的问题
2017/08/10 Javascript
分享ES6的7个实用技巧
2018/01/18 Javascript
基于jQuery实现的设置文本区域的光标位置
2018/06/15 jQuery
Flutter实现仿微信底部菜单栏功能
2019/09/18 Javascript
[01:15]《辉夜杯》北京网鱼队巡礼
2015/10/26 DOTA
在Python中封装GObject模块进行图形化程序编程的教程
2015/04/14 Python
深入理解Python中range和xrange的区别
2017/11/26 Python
Python贪心算法实例小结
2018/04/22 Python
使用Python的Dataframe取两列时间值相差一年的所有行方法
2018/07/10 Python
Python地图绘制实操详解
2019/03/04 Python
python中pytest收集用例规则与运行指定用例详解
2019/06/27 Python
解决torch.autograd.backward中的参数问题
2020/01/07 Python
python GUI库图形界面开发之PyQt5布局控件QHBoxLayout详细使用方法与实例
2020/03/06 Python
python pandas.DataFrame.loc函数使用详解
2020/03/26 Python
如何在windows下安装Pycham2020软件(方法步骤详解)
2020/05/03 Python
Python文件名匹配与文件复制的实现
2020/12/11 Python
学会迭代器设计模式,帮你大幅提升python性能
2021/01/03 Python
iframe在移动端的缩放的示例代码
2018/10/12 HTML / CSS
米兰必去买手店排行榜首位:Antonioli
2016/09/11 全球购物
Pam & Gela官网:美国性感前卫女装品牌
2018/07/19 全球购物
晚会邀请函范文
2014/01/24 职场文书
2014年大学生职业规划书:未来不是梦,只要勇敢冲!
2014/09/22 职场文书
杭州黄龙洞导游词
2015/02/10 职场文书