读书笔记「Python编程:从入门到实践」_11.测试函数
11.1 测试函数
要学习测试,得有要测试的代码。下面是一个简单的函数,它接受名和姓并返回整洁的姓名:
- def get_formatted_name(first, last):
- """Generate a neatly formatted full name."""
- full_name = first + ' ' + last
- return full_name.title()
为核实get_formatted_name() 像期望的那样工作,我们来编写一个使用这个函数的程序。
程序names.py让用户输入名和姓,并显示整洁的全名:
- from name_function import get_formatted_name
- print("Enter 'q' at any time to quit.")
- while True:
- first = input("\nPlease give me a first name: ")
- if first == 'q':
- break
- last = input("Please give me a last name: ")
- if last == 'q':
- break
- formatted_name = get_formatted_name(first, last)
- print("\tNeatly formatted name: " + formatted_name + '.')
我们可以在每次修改get_formatted_name() 后都进行测试:运行程序names.py,并输入像Janis Joplin 这样的姓名,但这太烦琐
我们可以利用单元测试函数,每次修改完元source以后,直接运行单元测试函数来判断程序是否正确
11.1.1 单元测试和测试用例
Python标准库中的模块unittest 提供了代码测试工具。
单元测试 用于核实函数的某个方面没有问题;
测试用例 是一组单元测试,这些单元测试一起核实函数在各种情形下的行为都符合要求。
良好的测试用例考虑到了函数可能收到的各种输入,包含针对所有这些情形的测试。
全覆盖式测试 用例包含一整套单元测试,涵盖了各种可能的函数使用方式。
11.1.2 可通过的测试
test_name_function.py
- import unittest
- from name_function import get_formatted_name
- #创建了一个名为NamesTestCase 的类,用于包含一系列针对get_formatted_name() 的单元测试。
- #最好让它看起来与要测试的函数相关,并包含字样Test
- #这个类必须继承unittest.TestCase 类
- class NamesTestCase(unittest.TestCase):
- """测试name_function.py"""
- #我们运行testname_function.py时,所有以test 打头的方法都将自动运行
- def test_first_last_name(self):
- """能够正确地处理像Janis Joplin这样的姓名吗?"""
- formatted_name = get_formatted_name('janis', 'joplin')
- #使用了unittest 类最有用的功能之一:一个断言 方法。断言方法用来核实得到的结果是否与期望的结果一致
- self.assertEqual(formatted_name, 'Janis Joplin')
- unittest.main()
- .
- ----------------------------------------------------------------------
- Ran 1 test in 0.002s
- OK
11.1.3 不能通过的测试
test_name_function.py
- import unittest
- from name_function import get_formatted_name
- #创建了一个名为NamesTestCase 的类,用于包含一系列针对get_formatted_name() 的单元测试。
- #最好让它看起来与要测试的函数相关,并包含字样Test
- #这个类必须继承unittest.TestCase 类
- class NamesTestCase(unittest.TestCase):
- """测试name_function.py"""
- #我们运行testname_function.py时,所有以test 打头的方法都将自动运行
- def test_first_last_middle_name(self):
- """能够正确地处理像Wolfgang Amadeus Mozart这样的姓名吗?"""
- formatted_name = get_formatted_name('wolfgang', 'mozart', 'amadeus')
- self.assertEqual(formatted_name, 'Wolfgang Amadeus Mozart')
- unittest.main()
- E
- ======================================================================
- ERROR: test_first_last_middle_name (__main__.NamesTestCase)
- 能够正确地处理像Wolfgang Amadeus Mozart这样的姓名吗?
- ----------------------------------------------------------------------
- Traceback (most recent call last):
- File "d:\40.勉強資料\python\test_name_function.py", line 11, in test_first_last_middle_name
- formatted_name = get_formatted_name('wolfgang', 'mozart', 'amadeus')
- TypeError: get_formatted_name() takes 2 positional arguments but 3 were given
- ----------------------------------------------------------------------
- Ran 1 test in 0.003s
- FAILED (errors=1)
11.1.4 测试未通过时怎么办
测试未通过时怎么办呢?如果你检查的条件没错,测试通过了意味着函数的行为是对的,
而测试未通过意味着你编写的新代码有错。因此,测试未通过时,不要修改测试,而应修复导致测试不能通过的代码:
检查刚对函数所做的修改,找出导致函数行为不符合预期的修改
name_function.py
- def get_formatted_name(first, last, middle=''):
- """生成整洁的姓名"""
- if middle:
- full_name = first + ' ' + middle + ' ' + last
- else:
- full_name = first + ' ' + last
- return full_name.title()
11.1.5 添加新测试
- import unittest
- from name_function import get_formatted_name
- #创建了一个名为NamesTestCase 的类,用于包含一系列针对get_formatted_name() 的单元测试。
- #最好让它看起来与要测试的函数相关,并包含字样Test
- #这个类必须继承unittest.TestCase 类
- class NamesTestCase(unittest.TestCase):
- """测试name_function.py"""
- #我们运行testname_function.py时,所有以test 打头的方法都将自动运行
- def test_first_last_name(self):
- """能够正确地处理像Janis Joplin这样的姓名吗?"""
- formatted_name = get_formatted_name('janis', 'joplin')
- #使用了unittest 类最有用的功能之一:一个断言 方法。断言方法用来核实得到的结果是否与期望的结果一致
- self.assertEqual(formatted_name, 'Janis Joplin')
- def test_first_last_middle_name(self):
- """能够正确地处理像Wolfgang Amadeus Mozart这样的姓名吗?"""
- formatted_name = get_formatted_name('wolfgang', 'mozart', 'amadeus')
- self.assertEqual(formatted_name, 'Wolfgang Amadeus Mozart')
- unittest.main()
11.2 测试类
11.2.1 各种断言方法
方法用途
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 中
11.2.2 一个要测试的类
- 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)
11.2.3 测试Anonymous
- import unittest
- from survey import AnonymousSurvey
- class TestAnonmyousSurvey(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_three_responses(self):
- """测试三个答案会被妥善地存储"""
- question = "What language did you first learn to speak?"
- my_survey = AnonymousSurvey(question)
- responses = ['English', 'Spanish', 'Mandarin']
- for response in responses:
- my_survey.store_response(response)
- for response in responses:
- self.assertIn(response, my_survey.responses)
- unittest.main()
11.2.4 方法setUp()
在前面的test_survey.py中,我们在每个测试方法中都创建了一个AnonymousSurvey 实例,并在每个方法中都创建了答案。
unittest.TestCase 类包含方法setUp() ,让我们只需创建这些对象一次,并在每个测试方法中使用它们。
如果你在TestCase 类中包含了方法setUp() ,Python将先运行它,再运行各个以test_打头的方法。这样,在你编写的每个测试方法中都可使用在方法setUp() 中创建的对象了
- import unittest
- from survey import AnonymousSurvey
- class TestAnonymousSurvey(unittest.TestCase):
- """针对AnonymousSurvey类的测试"""
- #可在setUp() 方法中创建一系列实例并设置它们的属性,再在测试方法中直接使用这些实例。
- #相比于在每个测试方法中都创建实例并设置其属性,这要容易得多
#方法setUp() 做了两件事情:创建一个调查对象;创建一个答案列表- def setUp(self):
- """
- 创建一个调查对象和一组答案,供使用的测试方法使用
- """
- question = "What language did you first learn to speak?"
- self.my_survey = AnonymousSurvey(question)
- self.responses = ['English', 'Spanish', 'Mandarin']
- 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_responses(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()
读书笔记「Python编程:从入门到实践」_11.测试函数的更多相关文章
- 读书笔记「Python编程:从入门到实践」_9.类
9.1 创建和使用类 面向对象编程——Object Oriented Programming,简称OOP,是一种程序设计思想. OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数. 把 ...
- 读书笔记「Python编程:从入门到实践」_10.文件和异常
10.1 从文件中读取数据 10.1.1 读取整个文件 with open(~) as object: contents=object.read() with open('C:/Users/jou/ ...
- 读书笔记「Python编程:从入门到实践」_8.函数
8.1 定义函数 def greet_user(): # def 来告诉Python你要定义一个函数.这是函数定义 """Hello World""& ...
- 读书笔记「Python编程:从入门到实践」_7.用户输入和while循环
7.1 函数input()的工作原理 函数input() 让程序暂停运行,等待用户输入一些文本.获取用户输入后,Python将其存储在一个变量中,以方便你使用. message = input(&qu ...
- 读书笔记「Python编程:从入门到实践」_6.字典
6.1 一个简单的字典 alien_0 = {'color': 'green', 'points': 5} print(alien_0['color']) print(alien_0['points' ...
- 读书笔记「Python编程:从入门到实践」_5.if语句
5.1 一个简单示例 cars = ['audi', 'bmw', 'subaru', 'toyota'] for car in cars: if car == 'bmw': print(car.up ...
- 读书笔记「Python编程:从入门到实践」_4.操作列表
4.1 遍历整个列表 4.1.1 深入地研究循环 4.1.2 在for循环中执行更多的操作 4.1.3 在for循环结束后执行一些操作 例 magicians = ['alice', ' ...
- 读书笔记「Python编程:从入门到实践」_3.列表简介
3.1 列表是什么 列表 由一系列按特定顺序排列的元素组成. 在Python中,用方括号([] )来表示列表,并用逗号来分隔其中的元素. 3.1.1 访问列表元素 指出列表的名称,再指出元素的索引 ...
- 读书笔记「Python编程:从入门到实践」_2.变量和简单数据类型
做了大半年RPA了,用的工具是Kapow. 工作没有那么忙,不想就这么荒废着,想学点什么.就Python吧. 为期三个月,希望能坚持下来. 2.1 变量的命名和使用 变量名只能包含字母.数字和下划线. ...
随机推荐
- 【codeforces 508B】Anton and currency you all know
[题目链接]:http://codeforces.com/contest/508/problem/B [题意] 给你一个奇数; 让你交换一次数字; 使得这个数字变成偶数; 要求偶数要最大; [题解] ...
- 【codeforces 514B】Han Solo and Lazer Gun
[题目链接]:http://codeforces.com/contest/514/problem/B [题意] 每次攻击可以把经过自己的一条直线上的所有点都毁掉; 然后给你n个目标物的坐标 问你最少要 ...
- Pycharm 的基本操作
下载:https://www.jetbrains.com/pycharm/ 安装:随意安装在那个目录都可以 注册:可以采用 激活码 或者激活服务器,并对应在选项下面填入激活码或者激活服务器URL即可. ...
- nyoj 310二分+dinic
#include<stdio.h> #include<queue> #include<string.h> using namespace std; #define ...
- 洛谷—— P2424 约数和
https://www.luogu.org/problem/show?pid=2424 题目背景 Smart最近沉迷于对约数的研究中. 题目描述 对于一个数X,函数f(X)表示X所有约数的和.例如:f ...
- Lucene5学习之使用MMSeg4j分词器
分类:程序语言|标签:C|日期: 2015-05-01 02:00:24 MMSeg4j是一款中文分词器,详细介绍如下: 1.mmseg4j 用 Chih-Hao Tsai 的 MMSeg 算法( ...
- 【Android开发VR实战】三.开发一个寻宝类VR游戏TreasureHunt
转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/53939303 本文出自[DylanAndroid的博客] [Android开发 ...
- [ACM] ZOJ 3819 Average Score (水题)
Average Score Time Limit: 2 Seconds Memory Limit: 65536 KB Bob is a freshman in Marjar Universi ...
- gradle打包android (实现外部导入签名文件、多渠道打包、导入ant脚本)
近期一直在做android自己主动打包,之前已经完毕了用纯命令行的形式打包.原生态ant脚本打包.和基于android的SDK的打包.而且实现了多渠道打包,后来同事推荐了gradle,网上的资料说gr ...
- Codeforces Round #332 (Div. 2)C. Day at the Beach 树状数组
C. Day at the Beach One day Squidward, Spongebob and Patrick decided to go to the beach. Unfortuna ...