单元测试(unit testing)
是指对软件中的最小可测试单元进行检查和验证。对于单元测试中单元的含义,一般来说,要根据实际情况去判定其具体含义,如C语言中单元指一个函数,Java里单元指一个类,图形化的软件中可以指一个窗口或一个菜单等。总的来说,单元就是人为规定的最小的单元测试是什么。
1.2unittest概念及运行原理
Unittest是python中的一个单元测试框架,要想了解unittest的运行原理首先来了解一下unittest中的几个概念。
Unittest有几个核心的组件:TestCase、TestSuite、TestLoader、 TestRunner、TestReport
Ø 测试用例(****Testcase****)
测试用例,测试用例里面会有很多测试方法,是单元测试中最小维度的测试行为,一个测试用例就是一个完整的测试单元,包括测试前准备环境的搭建(setUp),执行测试代码(run),以及测试后环境的还原(tearDown)
Ø 测试套件(TestSuite)
多个测试用例集合在一起,就是TestSuite,可以批量执行一个测试套件内所有的测试用例,而且TestSuite也可以嵌套TestSuite
Ø TestLoader
TestLoader是用来加载TestCase到TestSuite中的,其中有几个loadTestsFrom__()方法,就是从各个地方寻找TestCase,创建它们的实例,然后add到TestSuite中,再返回一个TestSuite实例
Ø Test fixture
对一个测试用例环境的搭建和销毁,是一个fixture,通过覆盖TestCase的setUp()和tearDown()方法来实现。用途是比如说在这个测试用例中需要访问数据库,那么可以在setUp()中建立数据库连接以及进行一些初始化,在tearDown()中清除在数据库中产生的数据,然后关闭连接。注意tearDown的过程很重要,要为以后的TestCase留下一个干净的环境。
Ø TextTestReport
测试的结果会保存到TextTestResult实例中,包括运行了多少测试用例,成功了多少,失败了多少等信息
上面几个概念之间的关系可用如下图表示:
首先是要写好TestCase,然后由TestLoader加载TestCase到TestSuite,然后由TextTestRunner来运行TestSuite,运行的结果保存在TextTestResult中,整个过程集成在unittest.main模块中。
注意:
Unittest中SetUp()方法在每条用例执行前都会执行
可以跳过执行测试用例,跳过整个类或单个的方法,@unittest.skip(类名或方法名)
每个测试方法以test开头,否则不被unittest识别
Unittest.main()默认根据ACSII码的顺序加载测试用例,数字与字母的顺序为:09,AZ,a~z
Ø 对于类来说,class TestAxx 会优先于class TestBxx被执行
Ø 对于方法来说,test_aaa()方法会有优先于test_bbb()被执行
如果想让用例按照指定的顺序执行,两种方式:
-
通过TestSuite按照顺序添加想要执行的方法
suite = unittest.TestSuite()
suite.addTest(TestBddClass("method_name_c"))
suite.addTest(TestBddClass("method_name_a"))
- 通过控制方法名字来实现
一般测试用例会比较多,建议采用第二种
- 如果想按照指定的顺序执行用例就用到了TestSuite(),
tests= [TestMathFunc("test_add"), TestMathFunc("test_minus"), TestMathFunc("test_divide")]
suite.addTests(tests)
TestLoader().loadTestsFromTestCase(TestLogin)添加到suite,然后执行
-
执行多个py文件的用例,方法是新建一个新的py文件,采用discover
test_dir = './case'
discover = unittest.defaultTestLoader.discover(test_dir, pattern='test*.py')
runner = unittest.TextTestRunner()
runner.run(discover)
将用例加载到suite中的方法有:
1)suite.addTest(TestLogin(‘test_login1’))
2)suite.addTests([TestLogin(‘test_login1’),TestLogin(‘test_login2’)])
3)# 用addTests + TestLoader
i.
#loadTestsFromName(),传入'模块名.TestCase名'
suite.addTests(unittest.TestLoader().loadTestsFromName('test_mathfunc.TestMathFunc'))
suite.addTests(unittest.TestLoader().loadTestsFromNames(['test_mathfunc.TestMathFunc']))
Ii.
#loadTestsFromTestCase(),传入TestCase
suite.addTests(unittest.TestLoader().loadTestsFromTestCase(TestMathFunc))
注意,用TestLoader的方法是无法对case进行排序的,同时,suite中也可以套suite。
- 解决数据依赖:可以用global、写到配置文件或数据库
9.一个类下的多个test方法指的是多个测试用例,之所以放在同一个类下是因为它们
1.Unittest.main()执行测试用例的顺序是不能人为控制的,根据case或类的名称执行的,那么要想按照我们指定的顺序执行,需要用到TestSuite
tests = [TestMathFunc("test_add"), TestMathFunc("test_minus"), TestMathFunc("test_divide")]
suite.addTests(tests)
2.suite可以一个一个添加用例进来,也可以一下子添加多个用例
a. 如果是一个一个添加进来就是:
suite = unittest.TestSuite()
suite.addTest(TestLogin('test_02'))
suite.addTest(TestLogin('test_01'))
unittest.TextTestRunner(verbosity=2).run(suite)
b. 添加多个用例到suite中执行
suite = unittest.TestSuite()
tests=[TestMathFunc("test_add"),TestMathFunc("test_minus"), TestMathFunc("test_divide")]
suite.addTests(tests)
runner = unittest.TextTestRunner(verbosity=2)
runner.run(suite)
- 如果想一个类一个类执行,但是又不想把该类下的每一个方法都添加进去,那么可以下面这样写,用TestLoader()将整个类添加到suite()中
三、TestLoader添加多个测试类,这种情况下是不能指定顺序的
suite1 = TestLoader().loadTestsFromTestCase(TestLogin2)
suite2 = TestLoader().loadTestsFromTestCase(TestLogin)suite = unittest.TestSuite(suite1)
添加多个测试集
suite = unittest.TestSuite([suite2,suite1])
runner = unittest.TextTestRunner()
runner.run(suite)
下面是一段没有运行通过的代码:
def test_muti_dirs1():
dir1 = 'case/case1'
dir2 = 'case/case2'
test_dirs = list()
test_dirs.append(dir1)
test_dirs.append(dir2)
# discover = unittest.defaultTestLoader.discover(dir1, pattern='test*.py')
# runner = unittest.TextTestRunner()
# runner.run(discover)
# print(test_dirs)
# for iterm in test_dirs:
# print(iterm)
for iterm in test_dirs:
unittest.defaultTestLoader._top_level_dir = None
discover = unittest.defaultTestLoader.discover(start_dir=iterm)
unittest.TextTestRunner().run(discover)