一文搞懂Python Unittest测试方法执行顺序
Unittest
unittest大家应该都不陌生。它作为一款博主在5-6年前最常用的单元测试框架,现在正被pytest,nose慢慢蚕食。
渐渐地,看到大家更多的讨论的内容从unittest+HTMLTestRunner变为pytest+allure2等后起之秀。
不禁感慨,终究是自己落伍了,跟不上时代的大潮了。
回到主题
感慨完了,回到正文。虽然unittest正在慢慢被放弃,但是它仍然是一款很全面的测试框架。
今天在群里看到有个群友的一番言论,激起了我的一番回忆。
自己以前是知道unittest的执行顺序并不是按照编写test方法的顺序执行,而是按照字典序执行的。但遗憾的是我都是投机取巧去解决的问题(后面会讲)。
下面我们就来探讨下unittest类的test方法的执行顺序问题。
源码初窥
研究一下源码(unittest.TestLoader)可以发现,在加载一个class下面的test方法的时候,原生Loader进行了排序,并且根据functools.cmp_to_key方法对测试方法列表进行了排序。
我们知道,unittest是不需要我们指定对应的方法,说白了,它是从类里面自动获取到咱们的方法,并约定了以test开头的方法都会被视为测试方法。
查询一下self.sortTestMethodsUsing(这个是一个排序的方式)。
可以看到这个比较方法写的很明确了,如果x < y那么返回-1,x = y则返回0,x > y返回1。
其实大家可能不知道Python里面的字符串也是可以比较的,在此必须说明一下字典序。我们来看看这个例子:
a = "abc"
b = "abcd"
c = "abce"
print(a > b)
print(b > c)
猜猜看执行结果,很显然,字典序的比较,是按A-Z的顺序来比较的,如果前缀一样但长度不一样,那么长度长的那个,字典序靠后。
了解了字典序以后,我们就不难知道,在unittest里面它寻找case的过程可以这样简化:
找到对应类下面以test开头的测试方法
对他们进行字典序排序
依次执行
这样就不难解释为什么我们有时候写的case不按照自己想的顺序来。
回到问题的本质
搞清楚为什么用例会乱,那就想到对应的解决方案。由于修改源码是不太合适的,那我们有2个策略去达成目的。
比如我有多个test方法:
class Testcase(unittest.TestCase):def setUp(self) -> None:passdef test_1(self):print("执行第一个")def test_2(self):print("第二个")def test_3(self):print("第三个")def test_10(self):print("第四个")def test_11(self):print("第五个")def tearDown(self) -> None:passif __name__ == "__main__":unittest.main()
执行起来,按照字典序,其实是1 10 11 2 3的顺序。
1. 以字典序的方式编写test方法
我们可以手动修改test方法的名称,这也是我早前的处理方式。也就是说把想要先执行的case字典序排到前面:
class Testcase(unittest.TestCase):def setUp(self) -> None:passdef test_0_1(self):print("执行第一个")def test_0_2(self):print("第二个")def test_0_3(self):print("第三个")def test_1_0(self):print("第四个")def test_1_1(self):print("第五个")def tearDown(self) -> None:pass
我们可以把数字按位数拆开,个位数就把10位补0,这样就能达到效果,如果会写100个case,我们就需要补2个0,比如0_0_1,当然一个文件里面也不会有太多case。
如果遇到test_login这种怎么办呢,不是数字结尾的方法。
其实是一样的,可以写成test_数字_业务的模式。番货写了一个装饰器专门解决这样的问题,大家可以去参考下。
2. 回归本质,从根本解决问题
方案1用了番货的装饰器,好是好,但是改变了方法本身的名称,我们其实可以针对他的排序方式入手,按照我们编写case的顺序排序测试方法,就能达到想要的目的。
说说思路:
- 手写一个loader继承自TestLoader类,改写里面的排序方法
- 在unittest运行的时候传入这个新的loader
来看看完整代码,注释里面写的很完善了。
import unittestclass MyTestLoader(unittest.TestLoader):def getTestCaseNames(self, testcase_class):# 调用父类的获取“测试方法”函数test_names = super().getTestCaseNames(testcase_class)# 拿到测试方法listtestcase_methods = list(testcase_class.__dict__.keys())# 根据list的索引对testcase_methods进行排序test_names.sort(key=testcase_methods.index)# 返回测试方法名称return test_namesclass Testcase(unittest.TestCase):def setUp(self) -> None:passdef test_1(self):print("执行第一个")def test_2(self):print("第二个")def test_3(self):print("第三个")def test_10(self):print("第四个")def test_11(self):print("第五个")def tearDown(self) -> None:passif __name__ == "__main__":unittest.main(testLoader=MyTestLoader())
执行了一下还是不对,是不是哪里出了什么问题呢?
是因为pycharm有一种默认的unittest的调试方法,我们要改成普通的方法去执行。
试试用控制台执行:
今天的内容就讲到这里了,看懂的记得给个赞哦~
最后: 可以关注公众号:伤心的辣条 ! 进去有许多资料共享!资料都是面试时面试官必问的知识点,也包括了很多测试行业常见知识,其中包括了有基础知识、Linux必备、Shell、互联网程序原理、Mysql数据库、抓包工具专题、接口测试工具、测试进阶-Python编程、Web自动化测试、APP自动化测试、接口自动化测试、测试高级持续集成、测试架构开发测试框架、性能测试、安全测试等。
如果我的博客对你有帮助、如果你喜欢我的博客内容,请 “点赞” “评论” “收藏” 一键三连哦!推荐软件测试交流学习群:914172719 里面会分享一些资深架构师录制的视频录像
好文推荐
转行面试,跳槽面试,软件测试人员都必须知道的这几种面试技巧!
面试经:一线城市搬砖!又面软件测试岗,5000就知足了…
面试官:工作三年,还来面初级测试?恐怕你的软件测试工程师的头衔要加双引号…
什么样的人适合从事软件测试工作?
那个准点下班的人,比我先升职了…
测试岗反复跳槽,跳着跳着就跳没了…
一文搞懂Python Unittest测试方法执行顺序相关推荐
- 一文搞懂 Python 的 import 机制
一.前言 希望能够让读者一文搞懂 Python 的 import 机制 1.什么是 import 机制? 通常来讲,在一段 Python 代码中去执行引用另一个模块中的代码,就需要使用 Python ...
- python语言语句快的标记是什么_一文搞懂Python程序语句
原标题:一文搞懂Python程序语句 程序流 Python 程序中常用的基本数据类型,包括: 内置的数值数据类型 Tuple 容器类型 String 容器类型 List 容器类型 自然的顺序是从页面或 ...
- python中gbk字符原因报错_不想再被鄙视?那就看进来! 一文搞懂 Python 2 字符编码...
原标题:不想再被鄙视?那就看进来! 一文搞懂 Python 2 字符编码 程序员都自视清高,觉得自己是创造者,经常鄙视不太懂技术的产品或者QA.可悲的是,程序员之间也相互鄙视,程序员的鄙视链流传甚广, ...
- 不想再被鄙视?那就看进来! 一文搞懂 Python 2 字符编码
程序员都自视清高,觉得自己是创造者,经常鄙视不太懂技术的产品或者QA.可悲的是,程序员之间也相互鄙视,程序员的鄙视链流传甚广,作为一个Python程序员,自然最关心的是下面这幅图啦 我们项目组一值使用 ...
- python运行程序-一文看懂python程序的执行过程
本文主要介绍的是python程序的执行过程,首先介绍的是编译过程,其次介绍的是过程图解及编译字节码,最后介绍了codeobject对象的属性,具体的跟随小编一起来了解一下. 一.编译过程概述 当我们执 ...
- 一文搞懂 Python 私有属性 私有方法
1. 场景定义 私有属性 是指在 Python 的面向对象开发过程中,对象的某些属性只想在对象的内部被使用,但不想在外部被访问到这些属性. 即:私有属性是对象不愿意公开的属性. 私有方法 是指在 Py ...
- python中row是什么意思_一文搞懂Python中的yield
关注公众号「Python七号」,及时 get Python 技能. yield 可以实现生成器,可以实现协程. 什么是生成器,什么是协程,如果还不了解,可以继续往下看,概念可以不懂,只要理解它的作用和 ...
- 一文搞懂python threading(多线程)模块
命令提示符如何打开并运行python文件链接 一. 1.active_count() , activeConut() 方法 功能 active_count() , activeConut() 返回处于 ...
- 一文搞懂Python中的所有数组数据类型
数组类型是各种编程语言中基本的数组结构了,本文来盘点下Python中各种"数组"类型的实现. list tuple array.array str bytes bytearray ...
最新文章
- string去掉后四位_数字黑洞(下):四位数的黑洞
- .bash_profile 写入时间格式YYYY-MM-DD HH24:MI:SS 时报错 not a valid identifier
- Tensorflow学习
- 职场潜规则:领导常说的10句话,学会3句以上你不会混的太差!
- Flume 1.7 源码分析(三)程序入口
- SQL Server ldf 丢失的数据库恢复
- 使用MapReduce将HDFS数据导入Mysql
- 【渝粤教育】 广东开放大学21秋期末考试劳动和社会保障法10682k2
- 2021-11-10如何快速部署Ceph分布式高可用集群
- java.sql.connection 长时间不使用_车子长时间停放不使用,要做到这几点!不然报废是小,安全是大...
- Wireshark-001基本设置
- [CTSC2018]假面(概率DP)
- 安装金山词霸2007
- 【热门主题】紫色梦幻天使win7主题
- 超全现代虚幻UE4素材网站整理
- 三菱FX3U-PLC 前馈+PID闭环调节实现液压同步控制(比例换向阀)
- 3DEC/PFC离散元入门篇
- input和output哪个是充电_Angular2中Input和Output用法及示例
- 将pdf转成图片时,文字没法显示
- HTML、CSS实现手风琴效果
热门文章
- python微信头像_转:20行Python代码给微信头像戴帽子
- 山西台达plc可编程控制器_(PLC)可编程控制器的编程语言你了解吗?不妨看看...
- seo关键词互点软件报价_舟山seo关键词优化软件
- 图像处理中各种边缘检测的微分算子简单比较(Sobel,Robert, Prewitt,Laplacian,Canny)
- C++面试题-指针-指针与指针的引用
- windows 安装 pip工具
- 堆、栈、堆栈详谈--复习
- margin与padding的bug
- 解析Java的JNI编程中的对象引用与内存泄漏问题
- 接口测试Fiddler实战