python unittest模块使用

学习完本篇,你将会

  • 编写python单元测试用例
  • 独立编写加减乘数的测试用例,同时满足一定条件可以跳过用例

简单认识单元测试

#例一:
#计算器类
Class count:
    def __init__(self,a,b):
	self.a = int(a)
	self.b = int(b)
#计算加法
    def add(self):
	return self.a+self.b
#测试类一:
#计算器类
Class TestCount:
     def test_add(self):
            try:
                   j=Count(2,3)
                   add=j.add()
                   assert(add == 5),’Interger addition result error’
             except ArrertioError as msg;
                    print msg
             else:
                     print ‘Test pass’
#执行测试类的测试方法
mytest = TestCount()
mytest.test_add()
#测试类二:
from calulator import Count
Import unittest
class TestCount(unittest.TestCase):
     def setUp(self):
            print “test start”
     
     def test_add(self):
             j = Count(2,3)
             self.assertEqual(j.add(),5)

      def tearDown(self):
             print “teset end”
If __name__ == ‘__main__’:
        unittest.main()
  • 通过比较上述的两种方法,可以看出测试类一虽然可以测试,单测试添加测试用例部方便(目前是尝试2,3,如果再添加需要增加很多代码),测试类二添加测试用例方面,可以随时添加。
  • 测试类二中,unittest可以直接把TestCount中以“test”命令的脚本直接识别为测试方法,并且执行它们
  • 通过也印证了在pycharm含有test命名的函数,执行并不是按照我们要求的顺序执行

unittest简介以及安装

  • 单元测试(unit testing):指对软件中的最小可测试单元进行检查和验证。对于单元测试中单元的含义,一般来说,要根据实际情况去判定其具体含义,单元就是人为规定的最小的被测功能模块。
  • Python单元测试框架(The Python unit testing framework):简称为PyUnit,是Kent Beck和Erich Gamma这两位所设计的 Junit 的Python版本。
  • unittest环境要求: PyUnit可以在Python 1.5.2及更高版本上运行。
  • unittest安装:python1.5.2以上的版本,python自带unittest模块,查看python是否包含unittest模块,通过D:\Python27\Lib\unittest,看是否含有unittest,如果没有需要下载安装,或者copy unittest模块。

Unittest中一些重要概论

  • Test Case:Test Case的实例就是一个测试用例。一个完成的是测试用例的流程包含:1、测试前环境的搭建(setUp)、实现测试过程的代码(run),以及测试后环境的还原(tearDown)。
  • Test Suit:验证一个功能往往需要多个用例,这些用例集合在一起叫测试套。Test Suit用来组装多个用例。可以通过addTest加载TestCase到TestSuite中,从而返回一个TestSuit实例。
  • Test Runner:测试执行也是单元测试中非常重要的一个概念,一般的单元测试框架都会提供一个丰富的执行策略和执行结果。在unittest单元测试框架中,通过TextTestRunner类提供的run()方法来执行test suit/test case。
  • Test Fixture:一个测试用例环境搭建和销毁,就是一个fixture。比如测试用例想访问数据库,可以通过setUp中连接数据库,在tearDown中关闭数据库即可。

Unittest例子

from calculator import count
import unittest
class TestCount(unittest.TestCase):
	def setUp(self):
		print “test start”
	def test_add1(self):
		j=Count(2,3)
		self.assertEqual(j.add(),5)
	def test_add2(self):
		j=Coun(47,76) 
		self.assertEqual(j.add(),117)
	def tearDown(self):
		print “test end”

If __name__ == ‘__main__’:
	suite = unittest.TestSuite()
	suite.addTest(TestCount(“test_add2”))
	runner = unittest.TextTestRunner()
	runner.run(suite)

调用Unittest框架的TestSuit()类来创建测试套件,通过它所提供的addTest()方法来添加测试用例测试test_add2()。调用

断言方法

方法

检查

assertEqual(a,b)

a=b

assertNoEqual(a,b)

a!=b

assertTrue(x)

bool(x) is True

assertFalse(x)

bool(x) is False

assertIs(a,b)

a is b

assertIsNot(a,b)

a is not b

assertIsNone

a is None

assertIsNoNone(x)

a is not None

assertIn(a,b)

a in b

assertNotIn(a,b)

a not in b

assertIsInstance(a,b)

isinstance(a,b)

assertIsInstance(a,b)

not is instance(a,b)

断言实例一

  • 语法:assertEqual(first,second,msg)
实例:
import unittest
class Test(unittest.TestCase):
     def setUp(self):
          num = input(“Enter a number:”)
          self.number = int(number)

     def test_case(self):
       self.assertEqual(self.number,10,msg=”your input is not 10”)

     def tearDown(self):
         pass

if __name__== “__main__”:
      unittest.main()

python unittest模块使用_Test

说明:用例先执行setUp前置条件,再执行测试用例test_case,再执行tearDown条件恢复。
arrertEqual:比较两个数的值是否相等,如果不相等输出msg中定义的提示信息,并执行以下结果:

断言实例二

  • 语法:assertTrue(表达式,msg)
实例:
#用于判断质数
def is_prime(n):
       if n<=1:
           return False
       for i in range(2,n):
            if n%i == 0:
                 return False
       return Ture

import unittest
class Test(unittest.TestCase):
     def setUp(self):
          self.number = int(number)

     def test_case(self):
       self.assertTrue(is_prime(7),msg=”Is not prime!”)

     def tearDown(self):
         pass

if __name__== “__main__”:
      unittest.main()

python unittest模块使用_测试用例_02

python unittest模块使用_Test_03

说明:用例先执行setUp前置条件,再执行测试用例test_case,再执行tearDown条件恢复。
arrertTrue:比较Bool是否为Ture,如果不是输出msg中定义的提示信息,并执行以下结果:

断言实例三

  • 语法:assertIn(first,second,msg=None)
实例:
#用于判断质数
import unittest
class Test(unittest.TestCase):
     def setUp(self):
          print “test start”

     def test_case(self):
        a = “hello”
        b = “hello world”
       self.assertIn(a,b,msg=”a is not in b”)

     def tearDown(self):
         print “test end”

if __name__== “__main__”:
      unittest.main()

python unittest模块使用_用例_04

python unittest模块使用_Test_05

unittest组织单元测试用例

例一

calulartor.py:
class Count():
    def __init__(self,a,b):
         self.a = int(a)
         self.b = int(b)

   #计算加法
   def add(self):
         return self.a+self.b

  #计算减法
   def sub(self):
         return self.a - self.b
test.py:
from calculator import Count
import unittest
class TestAdd(unittest.TestCase):
    def setUp(self):
         print “test add start”

   def test_add1(self):
          j = Count(2,3) 
          self.assertEqula(j.add(),5) 

   def test_add2(self):
          j = Count(41,76) 
          self.assertEqula(j.add(),117)  

   def tearDown(self):
         print “test add end”
class TestSub(unittest.TestCase):
         def setUp(self):
                print (“test sub start”)
         
         def test_sub1(self):
                 j = Count(2,3)
         
        def test_sub2(self):
                 j = Count(71,46) 
                 self.assertEqual(j.sub(),25)

         def tearDown(self):
                print “test sub end”
if __name__ == '__main__':
     #构造测试集
     suite = unittest.TestSuite()
     suite.addTest(TestAdd(“test_add1”))
     suite.addTest(TestAdd(“test_add2”))
     suite.addTest(TestSub(“test_sub1”))
     suite.addTest(TestSub(“test_sub2”))

     #运行测试集合

     runner =  unittest.TextTestRunner ()
     runner.run(suite)

TestSuit:TestSuit类中addTest()方法把不同测试类中的测试方法组装到测试套件中。通过运行测试套件,来执行所有的测试用例
通过运行可得知每次setUp以及tearDown都会被执行,如果用例很多初始化执行以及恢复条件都是一样,可以通过继承的方式可大大提高编写用例效率

例二

calulartor.py:
class Count():
    def __init__(self,a,b):
         self.a = int(a)
         self.b = int(b)

   #计算加法
   def add(self):
         return self.a+self.b

  #计算减法
   def sub(self):
         return self.a - self.b
test.py:
from calculator import Count
import unittest
class TestAdd(unittest.TestCase):
    def setUp(self):
         print “test add start”

   def test_add1(self):
          j = Count(2,3) 
          self.assertEqula(j.add(),5) 

   def test_add2(self):
          j = Count(41,76) 
          self.assertEqula(j.add(),117)  

   def tearDown(unittest.TestCase):
         print “test add end” 

class TestSub(unittest.TestCase):
         def setUp(self):
                print (“test sub start”)
         
         def test_sub1(self):
                 j = Count(2,3)
         
        def test_sub2(self):
                 j = Count(71,46) 
                 self.assertEqual(j.sub(),25)

         def tearDown(self):
                print “test sub end”
if __name__ == '__main__':
     #构造测试集
     suite1 = unittest.TestLoader().LoadTestsFromTestCase(TestAdd)     
     suite2 = unittest.TestLoader().LoadTestsFromTestCase(TestSub)                    
     suite = unittest.TestSuite([suite1,suite2])
     #运行测试集合
     unittest.TextTestRunner ()
     runner.run(suite)

TestLoader:用来加载TestCase到TestSuite中的,其中有几个 loadTestsFrom__()方法,就是从各个地方寻找TestCase,创建它们的实例,然后add到TestSuite中,再返回一个TestSuite实例;

discover自动发现用例

通过上述suite方式提供构造用例,如果用例数少,比较方便,但自动化设计一般情况下用例数都达到成百上千条用例,这个时候就比较麻烦。unittest提供一个单元测试框架自动识别测试用例,TestLoad类中提供了discover方法来自动识别测试用例。

  • 语法:discover(start_dir,pattern='test&.py',top_level_dir = None)
    1.start_dir:要测试的模块名或测试用例目录
    2.pattern='test.py':表示用例文件名的匹配原则。此处匹配文件名以“test”开头的“.py”类型的文件,星号“”表示任意多个字符
    3.top_level_dir = None:测试模块的顶层目录,如果没有顶层目录,默认为None。
runtest.py例子:
import unittest
#定义测试用例的目录为当前目录
test_dir ='./'
discover = unittest.defaultTestLoad.discover(test_dir,pattern = 'test*.py')

if __name__ == '__main__':
       runner = unittest.TextTestRunner()
       runner.run(discover)

用例执行顺序

test.py:
import unittest
class TestBdd(unittest.TestCase):
        def setUp(self):
               print “test TestBdd”

        def test_ccc(self):
               print “test ccc”

        def test_aaa(self):
               print “test aaa”

         def tearDown(self):
               pass

class TestAdd(unittest.TestCase):
         def setUp(self):
              print “test TestAdd:”
         
         def test_bbb(self):
              print “test bbb”

         def test_ddd(self):
              print “test   ddd”
         
         def tearDown(self):
               pass

if __name__ == '__main__':
         unittest.main()
通过unittest.TestSuite方法
if __name__ == '__main__':
     #构造测试集
     suite = unittest.TestSuite()
     suite.addTest(TestBdd(“test_ccc”))
     suite.addTest(TestAdd(“test_bbb”))
     suite.addTest(TestBdd(“test_aaa”))
     suite.addTest(TestAdd(“test_ddd”))

     #运行测试集合

     runner =  unittest.TestSuite()
     runner.run(suite)
  • 通过运行发现,使用unittest.mian()
    的方式,无论运行多少次都按照一定方式来执行,而且每次顺序都是一样,而通过unittest.TestSuite的方式是按照加载的顺序执行。
  • 通过默认的方式运行是按照ASCII的顺序加载测试用例,数字与字母的顺序为:0-9,A-Z,a-z。因此没有按照用例从上到下的顺序执行。而通过unittest.TestSuite是按照加载的顺序执行。

fixtures学习

test.py:
import unittest
def setUpModule():
      print “test module start.......”

def tearDownModule():
      print “test module end.......”

class Test(unittest.TestCase):
        @classmethod
         def setUpClass(cls):
                print “test class start.......”

        @classmethod
         def tearDownClass(cls):
                print “test class end.......”

        def setUp(self):
               print “test case...start”

       def tearDown(self):
               print “test case...end”

        def test_case1(self):
               print “test case1”

        def test_case2(self):
               print “test case2”
 
if __name__ == '__main__':
         unittest.main()
  • fixtures可以理解为夹心饼干外层的两片饼干,这里两片饼干就是setUp/tearDown,中间的心就是测试用例
  • setUpModule/tearDownModule:在整个模块的开始与结束时被执行
  • setUpClass/tearDownClass:在测试类开始与结束时被执行
  • setUp/tearDown:在测试用例的开始与结束时被执行

测试