• 首页 首页 icon
  • 工具库 工具库 icon
    • IP查询 IP查询 icon
  • 内容库 内容库 icon
    • 快讯库 快讯库 icon
    • 精品库 精品库 icon
    • 问答库 问答库 icon
  • 更多 更多 icon
    • 服务条款 服务条款 icon

自动化测试——unittest框架(单元测试)

武飞扬头像
小河鱼磨洋工
帮助1

目录

一、unittest框架解析

1.1unittest的5个重要概念

1.1测试用例的编写及代码

1.2断言

1.3用例的执行顺序

1.4测试用例综合管理框架

1.5HTML报告生成


参考博文

一、unittest框架解析

unittest 单元测试提供了创建测试用例,测试套件以及批量执行的方案, unittest 在安装pyhton 以后就直接自带了,直接import unittest 就可以使用。

 unittest 对程序最小模块(类或者方法)的一种敏捷化的测试。

单元测试:针对程序模块中进行正确性校验的测试工作(特点:应用中的最小可测单元、测试执行速度快、发现问题,更容易定位,一般由开发人员进行,也即是白盒测试)

在自动化测试中,我们不需要做白盒测试。利用单元测试框架,创建一个类,继承unittest的TestCase,这样可以把每个case看成是一个最小的单元, 由测试容器组织起来,直接执行,同时引入测试报告。

单元unittest框架:针对UI界面的功能进行自动化测试。不同于Java和Junit单元测试框架

unittest的5个重要概念

unittest 各组件的关系为:

学新通

  1. test fixture(测试固件):即测试环境的搭建。比如创建临时的数据库,文件和目录等。一般通过初始化(前置)和清理测试环境(结束)实现。
    1. setUp():测试用例执行前的准备:如环境、数据等
    2. setDown():测试环境的清理操作,如关闭浏览器等
  2. test case:测试用例(def test_add()),TestCase 是编写单元测试用例最常用的类:一个测试用例就是一个完成的测试单元,需执行setup()--->run()---->setDown()
  3. test suite:测试用例的集合集,TestSuite 是最常用的类
    1. addTest(): 把单个测试用例一个一个的放进测试用例集合里
    2. makeSuit(): 把一个类里的所有测试用例添加到一个测试套件里
    3. TestLoader(): 同上
    4. discover(): 把一个确定的文件夹下,以某种形式命名文件中的所有测试用例形成一个测试套件
  4. test runner:执行测试用例
  5. test loader:生成测试报告

unittest的用例规则:

  • 1、测试文件必须导包:import unittest
  • 2、测试类必须继承 类:unittest.TestCase
  • 3、测试方法必须以 test_开头

TestCase类中的几个特殊方法:

  • setUp():测试前的初始化工作(所有测试用例开始前都会执行的)
  • tearDown():测试后的清理工作
  • setUpClass():@classmethod,在所有测试方法运行前执行,只会执行一遍
  • tearDownClass():@classmethod,在所有测试方法运行结束后执行,只会执行一遍

1.1测试用例的编写及代码

类代码cal.py:

  1.  
    class Math1:
  2.  
    def __init__(self,a,b):
  3.  
    self.a = int(a)
  4.  
    self.b = int(b)
  5.  
     
  6.  
    def add(self):
  7.  
    return self.a self.b
  8.  
     
  9.  
    def sub(self):
  10.  
    return self.a - self.b

 测试代码:

  1.  
    from cal import Math1
  2.  
    import unittest
  3.  
    # from selenium import webdriver
  4.  
     
  5.  
    #创建测试用例的类TestMath,必须继承类unittest.TestCase
  6.  
    class TestMath(unittest.TestCase):
  7.  
    #----------------------测试前的初始化工作---------------------#
  8.  
    @classmethod #加上修饰,说明是类的修饰,只在类里执行,因为只有一个集合类,所以只执行一遍
  9.  
    def setUpClass(cls):
  10.  
    print("setUpClass")
  11.  
     
  12.  
    def setUp(self):
  13.  
    print("setUp") #两条测试用例都会执行,执行两次
  14.  
     
  15.  
    #----------------------测试用例的编写---------------------#
  16.  
    def test_add(self):
  17.  
    j = Math1(4,3)
  18.  
    self.assertEqual(j.add(),7 )
  19.  
     
  20.  
    def test_sub(self):
  21.  
    i = Math1(26,5)
  22.  
    self.assertEqual(i.sub(), 21)
  23.  
     
  24.  
    #----------------------测试后的结束工作---------------------#
  25.  
    def tearDown(self):
  26.  
    print("tearDown") #两条测试用例都会执行,执行两次
  27.  
     
  28.  
    @classmethod
  29.  
    def tearDownClass(cls):
  30.  
    print("tearDownClass")
  31.  
     
  32.  
    #----------------------测试用例的执行---------------------#
  33.  
     
  34.  
    if __name__ == "__main__":
  35.  
     
  36.  
    #构造测试集:放入所有测试用例 :此处表示由两个测试用例组成的测试集
  37.  
    suite = unittest.TestSuite()
  38.  
    suite.addTest(TestMath('test_add')) #将单个测试用例放入测试用例集合中
  39.  
    suite.addTest(TestMath('test_sub'))
  40.  
     
  41.  
    #执行测试用例
  42.  
    runner = unittest.TextTestRunner() #调用方法
  43.  
    runner.run(suite) #直接执行测试用例集
学新通

1.2断言

断言:测试用例实际结果与预期结果的对比

测试用例的要素:测试环境、测试步骤、输入、实际结果、预期结果

unittest 的单元测试库中常用的断言方法:

学新通


1.3用例的执行顺序

默认执行规则:测试类或测试方法的数字语字母顺序:0-9,A-Z,a-z(先执行完一个类中的方法,才会执行下一个类)

忽略测试用例的执行:@unittest.skip("skipping")

1.4测试用例综合管理框架

测试综合管理框架包括:

1.StartEnd.py : 开始和结束工作:setUp 和tearDown

2.cal.py:  加减法实现,整体功能模块

3.test_add.py:单个功能测试用例

4.test_sub.py:单个功能测试用例

5.run.py:用例执行

F:用例出现缺陷

.:成功

E:脚本中出现错误

1)StartEnd.py:

  1.  
    import unittest
  2.  
     
  3.  
    class setUp_tearDown(unittest.TestCase):
  4.  
    def setUp(self):
  5.  
    print("开始测试")
  6.  
     
  7.  
    def tearDown(self):
  8.  
    print("测试结束")

2)cal.py:  加减法实现,整体功能模块

  1.  
    class Math1:
  2.  
    def __init__(self,a,b):
  3.  
    self.a = int(a)
  4.  
    self.b = int(b)
  5.  
     
  6.  
    def add(self):
  7.  
    return self.a self.b
  8.  
     
  9.  
    def sub(self):
  10.  
    return self.a - self.b

3)test_add.py:单个功能测试用例

  1.  
    from cal import *
  2.  
    from StartEnd import *
  3.  
     
  4.  
    class Test_add(setUp_tearDown):
  5.  
     
  6.  
    def test_add(self):
  7.  
    j = Math1(5,9)
  8.  
    self.assertEqual(j.add(),14)
  9.  
     
  10.  
    def test_addError(self):
  11.  
    j = Math1(5,9)
  12.  
    self.assertNotEqual(j.add(),6)

4)test_sub.py:单个功能测试用例

  1.  
    from StartEnd import *
  2.  
    from cal import *
  3.  
     
  4.  
    class Test_sub(setUp_tearDown):
  5.  
     
  6.  
    def test_sub(self):
  7.  
    i = Math1(48,45)
  8.  
    self.assertEqual(i.sub(),3)

5)run.py

  1.  
    import unittest
  2.  
     
  3.  
    test_dir = './'
  4.  
     
  5.  
    #批量执行当前测试用例
  6.  
     
  7.  
    discovery = unittest.defaultTestLoader.discover(test_dir,pattern="test*.py")
  8.  
     
  9.  
    if __name__ == "__main__":
  10.  
    runner = unittest.TextTestRunner()
  11.  
    runner.run(discovery)

6)完整的测试代码:使用了unittest框架的脚本

  1.  
    from selenium import webdriver
  2.  
    import unittest
  3.  
    import time
  4.  
    import os
  5.  
    from selenium.common.exceptions import NoAlertPresentException
  6.  
    from selenium.common.exceptions import NoSuchElementException
  7.  
     
  8.  
    class Baidu1(unittest.TestCase):
  9.  
     
  10.  
    def setUp(self): #初始化环境,测试开始前执行的操作
  11.  
    print("-----setUp-----")
  12.  
    self.driver = webdriver.Chrome()
  13.  
    self.url = "https://www.百度.com/"
  14.  
    self.driver.maximize_window()
  15.  
    time.sleep(3)
  16.  
     
  17.  
    def tearDown(self): #测试结束后清理环境
  18.  
    print("--------tearDown--------")
  19.  
    self.driver.quit() #关闭浏览器
  20.  
     
  21.  
    #--------------测试用例 test_hao 和test_hao -------#
  22.  
    # 每个测试用例执行时都会执行开始和结束操作:setup
  23.  
    #忽略测试用例的执行: @unittest.skip("skipping")
  24.  
    def test_hao(self):
  25.  
    driver = self.driver
  26.  
    url = self.url
  27.  
    driver.get(url)
  28.  
    driver.find_element_by_link_text("hao123").click() #定位到百度页面的超链接“hao123”,也可以定位到“新闻”等其他超链接
  29.  
    time.sleep(6)
  30.  
     
  31.  
    def test_h百度(self):
  32.  
    driver = self.driver
  33.  
    url = self.url
  34.  
    driver.get(url)
  35.  
    # self.assertTrue("百度一下,你就知道" == driver.title, msg="不一致!!!")
  36.  
    driver.find_element_by_id("kw").send_keys("告白气球")
  37.  
    driver.find_element_by_id("su").submit()
  38.  
    time.sleep(3)
  39.  
    print(driver.title)
  40.  
    try:
  41.  
    self.assertNotEqual(driver.title,"告白气球_百度搜索",msg="实际结果与预期结果不一致")
  42.  
     
  43.  
    except:
  44.  
    self.saveScreenAsPhoto(driver,"haohao.png")
  45.  
    time.sleep(3)
  46.  
     
  47.  
    def saveScreenAsPhoto(self,driver,file_name):
  48.  
    if not os.path.exists("./images"):
  49.  
    os.makedirs("./images")
  50.  
    now = time.strftime("%Y%m%d-%H%M%S",time.localtime(time.time()))
  51.  
    driver.get_screenshot_as_file("./images/" now "-" file_name)
  52.  
     
  53.  
     
  54.  
    if __name__ == "__main__":
  55.  
    unittest.main() #执行所有测试用例(类中的所有方法)
学新通

unittest提供了全局的main()方法:使用它可以方便地将一个单元测试模块变成可以直接运行的测试脚本。main()方法搜索所有包含在该模块中以”test"命名的测试方法,并自动执行他们。

1)跑测试用例套件的步骤:此前提是要先把单个测试用例添加到套件里

  • 先生成测试套件: suite = creatSuit()
  • 得到跑测试套件的runner:runner = unittest.TextTestRunner(verbosity=2)
  • 用runner跑套件:runner.run(suite)
  1.  
    import unittest
  2.  
    from src0716 import test百度1
  3.  
    from src0716 import test百度2
  4.  
    def createsuite():
  5.  
    #addTest
  6.  
    suite = unittest.TestSuite()
  7.  
    suite.addTest(test百度1.Baidu1("test_hao")) #添加测试文件中的测试用例,此方法只能一个一个添加
  8.  
    suite.addTest(test百度1.Baidu1("test_h百度"))
  9.  
    suite.addTest(test百度2.Baidu2("test_hao"))
  10.  
    suite.addTest(test百度2.Baidu2("test_百度search"))
  11.  
    return suite
  12.  
    if __name__=="__main__":
  13.  
    suite = createsuite()
  14.  
    runner = unittest.TextTestRunner(verbosity=2)
  15.  
    runner.run(suite)
学新通

1.5makeSuite()、TestLoader()、discover()的应用

makeSuite:把一个测试用例类里的所有测试用例case组成测试套件TestSuite 

TestLoader :创建类和模块的测试套件,使TestLoader().loadTestsFromTestCase(TestClass) 来加载测试类。

discover :通过递归方式到其子目录中从指定的目录开始, 找到所有测试模块并返回包含它们对象的TestSuite ,然后加载与模式匹配唯一的测试文件,discover(dir,pattern,top_level_dir=None)

(1)把一个类里的测试用例添加到套件间里,代码如下:

  1.  
    import unittest,csv
  2.  
    import os,sys
  3.  
    import time
  4.  
    import test百度1
  5.  
    import test百度2
  6.  
    #手工添加案例到套件,
  7.  
    def createsuite():
  8.  
    suite = unittest.TestSuite()
  9.  
    #将测试用例加入到测试容器(套件)中
  10.  
    suite.addTest(unittest.makeSuite(test百度1.Baidu1))
  11.  
    suite.addTest(unittest.makeSuite(test百度2.Baidu2))
  12.  
    return suite
  13.  
     
  14.  
    '''
  15.  
    suite1 = unittest.TestLoader().loadTestsFromTestCase(test百度1.Baidu1)
  16.  
    suite2 = unittest.TestLoader().loadTestsFromTestCase(test百度2.Baidu2)
  17.  
    suite = unittest.TestSuite([suite1, suite2])
  18.  
    return suite
  19.  
    '''
  20.  
     
  21.  
    if __name__=="__main__":
  22.  
    suite=createsuite()
  23.  
    runner = unittest.TextTestRunner(verbosity=2)
  24.  
    runner.run(suite)
学新通

(2)把一个文件夹下以某种形式命名(正则表达式匹配)的所有文件的测试用例都加载到测试套件:

  1.  
    import unittest,csv
  2.  
    import os,sys
  3.  
    import time
  4.  
     
  5.  
    #手工添加案例到套件,
  6.  
    def createsuite():
  7.  
    discover=unittest.defaultTestLoader.discover('../test',pattern='test*.py',top_le
  8.  
    vel_dir=None)
  9.  
    print discover
  10.  
    return discover
  11.  
     
  12.  
    if __name__=="__main__":
  13.  
    suite=createsuite()
  14.  
    runner = unittest.TextTestRunner(verbosity=2)
  15.  
    runner.run(suite)
学新通

1.6HTML报告生成

1)添加HTMLTestRunner.py文件:C:\Users\HY\AppData\Local\Programs\Python\Python38\lib

2)HTML报告的生成步骤:

解决HTML文件存放问题--->创建文件夹

HTML报告命名问题(动态命名,使每个文件名不一样)

  1.  
    # -*- coding: utf-8 -*-
  2.  
    import unittest,csv
  3.  
    import os,sys
  4.  
    import time
  5.  
    import HTMLTestRunner
  6.  
     
  7.  
    #手工添加案例到套件,
  8.  
    def createsuite():
  9.  
    discover=unittest.defaultTestLoader.discover('../test',pattern='test*.py',top_level_dir=None)
  10.  
    print(discover)
  11.  
    return discover
  12.  
     
  13.  
    if __name__=="__main__":
  14.  
    curpath=sys.path[0]
  15.  
    #取当前时间
  16.  
    now=time.strftime("%Y-%m-%d-%H %M %S",time.localtime(time.time()))
  17.  
    if not os.path.exists(curpath '/resultreport'):
  18.  
    os.makedirs(curpath '/resultreport')
  19.  
    filename=curpath '/resultreport/' now 'resultreport.html'
  20.  
    with open(filename,'wb') as fp:
  21.  
    #出html报告
  22.  
    runner = HTMLTestRunner.HTMLTestRunner(stream=fp,title=u'测试报告',description=u'用例执行情况',verbosity=2)
  23.  
    suite=createsuite()
  24.  
    runner.run(suite)
学新通

1.7错误截图

关键语句:driver.get_screenshot_as_file()    

  1.  
    def savescreenshot(self,driver,file_name):
  2.  
    if not os.path.exists('./image'):
  3.  
    os.makedirs('./image')
  4.  
    now=time.strftime("%Y%m%d-%H%M%S",time.localtime(time.time()))
  5.  
    #截图保存
  6.  
    driver.get_screenshot_as_file('./image/' now '-' file_name)
  7.  
    time.sleep

1.8数据驱动

@data :支持一个或者多个数据的传入

@unpack :多个数据传入需对一组数据进行映射,包括:测试数据和测试方法的映射

@file_data:测试数据在json文件中,和测试方法中的参数进行映射

这篇好文章是转载于:学新通技术网

  • 版权申明: 本站部分内容来自互联网,仅供学习及演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,请提供相关证据及您的身份证明,我们将在收到邮件后48小时内删除。
  • 本站站名: 学新通技术网
  • 本文地址: /boutique/detail/tanhfhajeh
系列文章
更多 icon
同类精品
更多 icon
继续加载