编写函数和类时,还可以为其编写测试.通过测试,可确定代码面对各种输出都能够按要求的那样工作.在程序中添加新的代码时,你也可以对其进行测试,确定它们不会破坏程序既有的行为.程序员都会犯错,因此每个程序员都必须经常测试其代码,在用户发现问题前找出它们.
学习目标:
学习如何使用Python模块unittest中的工具来测试代码.
学习编写测试用例,核实一系列输入都将得到预期的输出
要测试函数,得有要测试的代码.然后,大家也都会运行程序,然后进行相应的操作,再根据程序做出的应答判断程序是否正常.但不可置否,这个步骤太繁琐了.所幸python提供了一种自动测试函数输出的高效方式.
def get_formatted_name(first,last):
"""生成整洁的姓名"""
full_name = first + ' ' + last
return full_name.title()
下面是仅包含一个方法的测试用例,用于检查get_formtted_name()在给定名和姓时,能否正常地工作:
test_name_function.py
import unittest
from name_function import get_formatted_name
class NamesTestCase(unittest.TestCase):
"""测试name_function.py"""
def test_first_last_name(self):
"""能够正确地处理像Janis Joplin这样的姓名吗?"""
formatted_name = get_formatted_name('janis','joplin')
self.assertEqual(formatted_name,'Janis Joplin')
unittest.main()
运行结果:
.
----------------------------------------------------------------------
Ran 1 test in 0.001s
OK
self.assertEqual(formatted_name,'Janis Joplin')
def get_formatted_name(first,middle,last):
"""生成整洁的姓名"""
full_name = first + ' ' + middle + ' ' + last
return full_name.title()
再运行test_name_function.py程序,有:
E
======================================================================
ERROR: test_first_last_name (__main__.NamesTestCase)
能够正确地处理像Janis Joplin这样的姓名吗?
----------------------------------------------------------------------
Traceback (most recent call last):
File "d:\vscode\test.py", line 10, in test_first_last_name
formatted_name = get_formatted_name('janis','joplin')
TypeError: get_formatted_name() missing 1 required positional argument: 'last'
----------------------------------------------------------------------
Ran 1 test in 0.002s
FAILED (errors=1)
让我们来分析一下:
def get_formatted_name(first,last,middle=''):
"""生成整洁的姓名"""
if middle:
full_name = first + ' ' + middle + ' ' + last
else:
full_name = first + ' ' + last
return full_name.title()
以及在类NamesTestCase中增加一个新的测试单元
import unittest
from name_function import get_formatted_name
class NamesTestCase(unittest.TestCase):
"""测试name_function.py"""
def test_first_last_name(self):
"""能够正确地处理像Janis Joplin这样的姓名吗?"""
formatted_name = get_formatted_name('janis','joplin')
self.assertEqual(formatted_name,'Janis Joplin')
def test_first_middle_last_name(self):
"""能正常处理像Wolfgang Amadeus Mozart这样的姓名吗?"""
formatted_name = get_formatted_name('Wolfgang','Mozart','Amadeus')
self.assertEqual(formatted_name,'Wolfgang Amadeus Mozart')
unittest.main()
运行结果:
..
----------------------------------------------------------------------
Ran 2 tests in 0.001s
OK
现在,两个测试用例都通过了.这让我们深信这个函数既能正确地处理像Janis Joplin这样的姓名,也能正确到处理像Wolfgang Amadeus Mozart这样的姓名.
在前面,我们编写了针对单个函数的的测试,下面来编写针对类的测试
下面列表中描述了6个常用的断言方法:
方法 | 用途 |
---|---|
assertEqual(a,b) | 核实a == b |
assertNotEqual(a,b) | 核实a != b |
assertTrue(x) | 核实x为True |
assertFalse(x) | 核实x为False |
assertIn(item,list) | 核实item在list |
assertNotIn(item,list) | 核实item不在list中 |
class AnonymousSurvey():
"""收集匿名调查问卷的答案"""
def __init__(self,question):
"""存储一个问题,并为存储答案做准备"""
self.question = question
self.responses = []
def show_question(self):
"""显示调查问卷"""
print(self.question)
def store_response(self,new_response):
"""存储单份调查答案"""
self.responses.append(new_response)
def show_results(self):
"""显示收集到的所有答卷"""
print("Survey results:")
for response in self.responses:
print('- ' + response)
接下来编写一个测试,对AnonymousSurvey类的行为的一个方面进行验证:如果用户面对调查问题时只提供一个答案以及提供多个答案时,程序能够妥善将其存储:
import unittest
from survey import AnonymousSurvey
class TestAnonymousSurvey(unittest.TestCase):
"""针对AnonymousSurvey类的测试"""
def test_store_single_response(self):
"""测试单个答案能否被正常存储"""
question = "What language did you first learn to speak?"
my_survey = AnonymousSurvey(question)
my_survey.store_response('English')
self.assertIn('English',my_survey.responses)
def test_store_single_response(self):
"""测试多个个答案能否被正常存储"""
question = "What language did you first learn to speak?"
my_survey = AnonymousSurvey(question)
responses = ['English','Chinese','English']
for response in responses:
my_survey.store_response(response)
for response in responses:
self.assertIn(response,my_survey.responses)
unittest.main()
运行结果:
..
----------------------------------------------------------------------
Ran 2 tests in 0.001s
OK
前面的做法效果很好,但这些代码有重复的地方,下面将解决这一问题
import unittest
from survey import AnonymousSurvey
class TestAnonymousSurvey(unittest.TestCase):
"""针对AnonymousSurvey类的测试"""
def setUp(self):
"""
创建一个调查对象和一组答案,供使用的测试方法使用
"""
question = "What language did you first learn to speak?"
self.my_survey = AnonymousSurvey(question)
self.responses = ['English','Chinese','English']
def test_store_single_response(self):
"""测试单个答案能否被正常存储"""
self.my_survey.store_response(self.responses[0])
self.assertIn(self.responses[0],self.my_survey.responses)
def test_store_three_response(self):
"""测试多个个答案能否被正常存储"""
for response in self.responses:
self.my_survey.store_response(response)
for response in self.responses:
self.assertIn(response,self.my_survey.responses)
unittest.main()
运行结果:
..
----------------------------------------------------------------------
Ran 2 tests in 0.002s
OK
注意:运行测试用例时,每完成一个单元测试,python都会打印一个字符:测试通过时打印一个句点;测试引发错误时打印一个E;测试导致断言失败时打印一个F.这就是你运行测试用例时,在输出的第一行看到的句点和字符数量各不相同的原因.如果测试用例包含很多单元测试,需要运行很长时间,就可以通过这些结果来获悉有多少个测试通过了.
个人从书本学到的知识,作为自己学习笔记的同时,与各位朋友分享,如有不足,请多多指教
参考文献:《Python编程从入门到实践》【美】Eric Matthes 著 袁国忠 译