11.1 测试函数

  要学习测试,得有要测试的代码。下面是一个简单的函数,它接受名和姓并返回整洁的姓名:

  1. def get_formatted_name(first, last):
  2. """Generate a neatly formatted full name."""
  3. full_name = first + ' ' + last
  4. return full_name.title()

  为核实get_formatted_name() 像期望的那样工作,我们来编写一个使用这个函数的程序。

  程序names.py让用户输入名和姓,并显示整洁的全名:

  1. from name_function import get_formatted_name
  2. print("Enter 'q' at any time to quit.")
  3. while True:
  4. first = input("\nPlease give me a first name: ")
  5. if first == 'q':
  6. break
  7. last = input("Please give me a last name: ")
  8. if last == 'q':
  9. break
  10. formatted_name = get_formatted_name(first, last)
  11. 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

  1. import unittest
  2. from name_function import get_formatted_name
  3.  
  4. #创建了一个名为NamesTestCase 的类,用于包含一系列针对get_formatted_name() 的单元测试。
  5. #最好让它看起来与要测试的函数相关,并包含字样Test
  6. #这个类必须继承unittest.TestCase 类
  7. class NamesTestCase(unittest.TestCase):
  8. """测试name_function.py"""
  9. #我们运行testname_function.py时,所有以test 打头的方法都将自动运行
  10. def test_first_last_name(self):
  11. """能够正确地处理像Janis Joplin这样的姓名吗?"""
  12. formatted_name = get_formatted_name('janis', 'joplin')
  13. #使用了unittest 类最有用的功能之一:一个断言 方法。断言方法用来核实得到的结果是否与期望的结果一致
  14. self.assertEqual(formatted_name, 'Janis Joplin')
  15.  
  16. unittest.main()
  1. .
  2. ----------------------------------------------------------------------
  3. Ran 1 test in 0.002s
  4.  
  5. OK

  11.1.3 不能通过的测试

  test_name_function.py

  1. import unittest
  2. from name_function import get_formatted_name
  3.  
  4. #创建了一个名为NamesTestCase 的类,用于包含一系列针对get_formatted_name() 的单元测试。
  5. #最好让它看起来与要测试的函数相关,并包含字样Test
  6. #这个类必须继承unittest.TestCase 类
  7. class NamesTestCase(unittest.TestCase):
  8. """测试name_function.py"""
  9. #我们运行testname_function.py时,所有以test 打头的方法都将自动运行
  10. def test_first_last_middle_name(self):
  11. """能够正确地处理像Wolfgang Amadeus Mozart这样的姓名吗?"""
  12. formatted_name = get_formatted_name('wolfgang', 'mozart', 'amadeus')
  13. self.assertEqual(formatted_name, 'Wolfgang Amadeus Mozart')
  14.  
  15. unittest.main()
  1. E
  2. ======================================================================
  3. ERROR: test_first_last_middle_name (__main__.NamesTestCase)
  4. 能够正确地处理像Wolfgang Amadeus Mozart这样的姓名吗?
  5. ----------------------------------------------------------------------
  6. Traceback (most recent call last):
  7. File "d:\40.勉強資料\python\test_name_function.py", line 11, in test_first_last_middle_name
  8. formatted_name = get_formatted_name('wolfgang', 'mozart', 'amadeus')
  9. TypeError: get_formatted_name() takes 2 positional arguments but 3 were given
  10.  
  11. ----------------------------------------------------------------------
  12. Ran 1 test in 0.003s
  13.  
  14. FAILED (errors=1)

  11.1.4 测试未通过时怎么办

  测试未通过时怎么办呢?如果你检查的条件没错,测试通过了意味着函数的行为是对的,
  而测试未通过意味着你编写的新代码有错。因此,测试未通过时,不要修改测试,而应修复导致测试不能通过的代码:
  检查刚对函数所做的修改,找出导致函数行为不符合预期的修改

 name_function.py

  1. def get_formatted_name(first, last, middle=''):
  2. """生成整洁的姓名"""
  3. if middle:
  4. full_name = first + ' ' + middle + ' ' + last
  5. else:
  6. full_name = first + ' ' + last
  7. return full_name.title()

  11.1.5 添加新测试

  1. import unittest
  2. from name_function import get_formatted_name
  3.  
  4. #创建了一个名为NamesTestCase 的类,用于包含一系列针对get_formatted_name() 的单元测试。
  5. #最好让它看起来与要测试的函数相关,并包含字样Test
  6. #这个类必须继承unittest.TestCase 类
  7. class NamesTestCase(unittest.TestCase):
  8. """测试name_function.py"""
  9. #我们运行testname_function.py时,所有以test 打头的方法都将自动运行
  10. def test_first_last_name(self):
  11. """能够正确地处理像Janis Joplin这样的姓名吗?"""
  12. formatted_name = get_formatted_name('janis', 'joplin')
  13. #使用了unittest 类最有用的功能之一:一个断言 方法。断言方法用来核实得到的结果是否与期望的结果一致
  14. self.assertEqual(formatted_name, 'Janis Joplin')
  15.  
  16. def test_first_last_middle_name(self):
  17. """能够正确地处理像Wolfgang Amadeus Mozart这样的姓名吗?"""
  18. formatted_name = get_formatted_name('wolfgang', 'mozart', 'amadeus')
  19. self.assertEqual(formatted_name, 'Wolfgang Amadeus Mozart')
  20. 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 一个要测试的类

  1. class AnonymousSurvey():
  2. """收集匿名调查问卷的答案"""
  3. def __init__(self, question):
  4. """存储一个问题,并为存储答案做准备"""
  5. self.question = question
  6. self.responses = []
  7. def show_question(self):
  8. """显示调查问卷"""
  9. print(self.question)
  10. def store_response(self, new_response):
  11. """存储单份调查答卷"""
  12. self.responses.append(new_response)
  13. def show_results(self):
  14. """显示收集到的所有答卷"""
  15. print("Survey results:")
  16. for response in self.responses:
  17. print('- ' + response)

  11.2.3 测试Anonymous

  1. import unittest
  2. from survey import AnonymousSurvey
  3.  
  4. class TestAnonmyousSurvey(unittest.TestCase):
  5. """针对AnonymousSurvey类的测试"""
  6. def test_store_single_response(self):
  7. """测试单个答案会被妥善地存储"""
  8. question = "What language did you first learn to speak?"
  9. my_survey = AnonymousSurvey(question)
  10. my_survey.store_response('English')
  11. self.assertIn('English', my_survey.responses)
  12.  
  13. def test_store_three_responses(self):
  14. """测试三个答案会被妥善地存储"""
  15. question = "What language did you first learn to speak?"
  16. my_survey = AnonymousSurvey(question)
  17. responses = ['English', 'Spanish', 'Mandarin']
  18. for response in responses:
  19. my_survey.store_response(response)
  20. for response in responses:
  21. self.assertIn(response, my_survey.responses)
  22.  
  23. unittest.main()

  11.2.4 方法setUp()  

  在前面的test_survey.py中,我们在每个测试方法中都创建了一个AnonymousSurvey 实例,并在每个方法中都创建了答案。

  unittest.TestCase 类包含方法setUp() ,让我们只需创建这些对象一次,并在每个测试方法中使用它们。

  如果你在TestCase 类中包含了方法setUp() ,Python将先运行它,再运行各个以test_打头的方法。这样,在你编写的每个测试方法中都可使用在方法setUp() 中创建的对象了

  1. import unittest
  2. from survey import AnonymousSurvey
  3.  
  4. class TestAnonymousSurvey(unittest.TestCase):
  5. """针对AnonymousSurvey类的测试"""
  6. #可在setUp() 方法中创建一系列实例并设置它们的属性,再在测试方法中直接使用这些实例。
  7. #相比于在每个测试方法中都创建实例并设置其属性,这要容易得多
    #方法setUp() 做了两件事情:创建一个调查对象;创建一个答案列表
  8. def setUp(self):
  9. """
  10. 创建一个调查对象和一组答案,供使用的测试方法使用
  11. """
  12. question = "What language did you first learn to speak?"
  13. self.my_survey = AnonymousSurvey(question)
  14. self.responses = ['English', 'Spanish', 'Mandarin']
  15. def test_store_single_response(self):
  16. """测试单个答案会被妥善地存储"""
  17. self.my_survey.store_response(self.responses[0])
  18. self.assertIn(self.responses[0], self.my_survey.responses)
  19. def test_store_three_responses(self):
  20. """测试三个答案会被妥善地存储"""
  21. for response in self.responses:
  22. self.my_survey.store_response(response)
  23. for response in self.responses:
  24. self.assertIn(response, self.my_survey.responses)
  25. unittest.main()

读书笔记「Python编程:从入门到实践」_11.测试函数的更多相关文章

  1. 读书笔记「Python编程:从入门到实践」_9.类

    9.1 创建和使用类 面向对象编程——Object Oriented Programming,简称OOP,是一种程序设计思想. OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数. 把 ...

  2. 读书笔记「Python编程:从入门到实践」_10.文件和异常

    10.1 从文件中读取数据  10.1.1 读取整个文件 with open(~) as object: contents=object.read() with open('C:/Users/jou/ ...

  3. 读书笔记「Python编程:从入门到实践」_8.函数

    8.1 定义函数 def greet_user(): # def 来告诉Python你要定义一个函数.这是函数定义 """Hello World""& ...

  4. 读书笔记「Python编程:从入门到实践」_7.用户输入和while循环

    7.1 函数input()的工作原理 函数input() 让程序暂停运行,等待用户输入一些文本.获取用户输入后,Python将其存储在一个变量中,以方便你使用. message = input(&qu ...

  5. 读书笔记「Python编程:从入门到实践」_6.字典

    6.1 一个简单的字典 alien_0 = {'color': 'green', 'points': 5} print(alien_0['color']) print(alien_0['points' ...

  6. 读书笔记「Python编程:从入门到实践」_5.if语句

    5.1 一个简单示例 cars = ['audi', 'bmw', 'subaru', 'toyota'] for car in cars: if car == 'bmw': print(car.up ...

  7. 读书笔记「Python编程:从入门到实践」_4.操作列表

    4.1 遍历整个列表   4.1.1 深入地研究循环   4.1.2 在for循环中执行更多的操作   4.1.3 在for循环结束后执行一些操作  例 magicians = ['alice', ' ...

  8. 读书笔记「Python编程:从入门到实践」_3.列表简介

    3.1 列表是什么 列表 由一系列按特定顺序排列的元素组成. 在Python中,用方括号([] )来表示列表,并用逗号来分隔其中的元素. 3.1.1 访问列表元素 指出列表的名称,再指出元素的索引   ...

  9. 读书笔记「Python编程:从入门到实践」_2.变量和简单数据类型

    做了大半年RPA了,用的工具是Kapow. 工作没有那么忙,不想就这么荒废着,想学点什么.就Python吧. 为期三个月,希望能坚持下来. 2.1 变量的命名和使用 变量名只能包含字母.数字和下划线. ...

随机推荐

  1. 【codeforces 508B】Anton and currency you all know

    [题目链接]:http://codeforces.com/contest/508/problem/B [题意] 给你一个奇数; 让你交换一次数字; 使得这个数字变成偶数; 要求偶数要最大; [题解] ...

  2. 【codeforces 514B】Han Solo and Lazer Gun

    [题目链接]:http://codeforces.com/contest/514/problem/B [题意] 每次攻击可以把经过自己的一条直线上的所有点都毁掉; 然后给你n个目标物的坐标 问你最少要 ...

  3. Pycharm 的基本操作

    下载:https://www.jetbrains.com/pycharm/ 安装:随意安装在那个目录都可以 注册:可以采用 激活码 或者激活服务器,并对应在选项下面填入激活码或者激活服务器URL即可. ...

  4. nyoj 310二分+dinic

    #include<stdio.h> #include<queue> #include<string.h> using namespace std; #define ...

  5. 洛谷—— P2424 约数和

    https://www.luogu.org/problem/show?pid=2424 题目背景 Smart最近沉迷于对约数的研究中. 题目描述 对于一个数X,函数f(X)表示X所有约数的和.例如:f ...

  6. Lucene5学习之使用MMSeg4j分词器

      分类:程序语言|标签:C|日期: 2015-05-01 02:00:24 MMSeg4j是一款中文分词器,详细介绍如下: 1.mmseg4j 用 Chih-Hao Tsai 的 MMSeg 算法( ...

  7. 【Android开发VR实战】三.开发一个寻宝类VR游戏TreasureHunt

    转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/53939303 本文出自[DylanAndroid的博客] [Android开发 ...

  8. [ACM] ZOJ 3819 Average Score (水题)

    Average Score Time Limit: 2 Seconds      Memory Limit: 65536 KB Bob is a freshman in Marjar Universi ...

  9. gradle打包android (实现外部导入签名文件、多渠道打包、导入ant脚本)

    近期一直在做android自己主动打包,之前已经完毕了用纯命令行的形式打包.原生态ant脚本打包.和基于android的SDK的打包.而且实现了多渠道打包,后来同事推荐了gradle,网上的资料说gr ...

  10. 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 ...