前言

学pytest就不得不说fixture,fixture是pytest的精髓所在,就像unittest中的setup和teardown一样,如果不学fixture那么使用pytest和使用unittest是没什么区别的(个人理解)。

fixture用途

1.做测试前后的初始化设置,如测试数据准备,链接数据库,打开浏览器等这些操作都可以使用fixture来实现

2.测试用例的前置条件可以使用fixture实现

3.支持经典的xunit fixture ,像unittest使用的setup和teardown

4.fixture可以实现unittest不能实现的功能,比如unittest中的测试用例和测试用例之间是无法传递参数和数据的,但是fixture却可以解决这个问题

fixture定义

fixture通过@pytest.fixture()装饰器装饰一个函数,那么这个函数就是一个fixture,看个实例

# test_fixture.py

import pytest

@pytest.fixture()

def fixtureFunc():

return 'fixtureFunc'

def test_fixture(fixtureFunc):

print('我调用了{}'.format(fixtureFunc))

if __name__=='__main__':

pytest.main(['-v', 'test_fixture.py'])

执行结果

test_fixture.py .我调用了fixtureFunc

[100%]

========================== 1 passed in 0.02 seconds ===========================

Process finished with exit code 0

fixtureFunc 这个函数就是一个fixture,fixture函数内部可以实现一些初始化操作!

fixture使用

调用fixture有三种方式

方式1

fixture的名字直接作为测试用例的参数,上面的实例就这这种方式,再来看一个实例

# test_fixture.py

import pytest

@pytest.fixture()

def fixtureFunc():

return 'fixtureFunc'

def test_fixture(fixtureFunc):

print('我调用了{}'.format(fixtureFunc))

class TestFixture(object):

def test_fixture_class(self, fixtureFunc):

print('在类中使用fixture "{}"'.format(fixtureFunc))

if __name__=='__main__':

pytest.main(['-v', 'test_fixture.py'])

方式2

每个函数或者类前使用@pytest.mark.usefixtures('fixture')装饰器装饰

实例

# test_fixture.py

import pytest

@pytest.fixture()

def fixtureFunc():

print('\n fixture->fixtureFunc')

@pytest.mark.usefixtures('fixtureFunc')

def test_fixture():

print('in test_fixture')

@pytest.mark.usefixtures('fixtureFunc')

class TestFixture(object):

def test_fixture_class(self):

print('in class with text_fixture_class')

if __name__=='__main__':

pytest.main(['-v', 'test_fixture.py'])

方式3

指定fixture的参数autouse=True这样每个测试用例会自动调用fixture(其实这里说的不是很准确,因为还涉及到fixture的作用范围,那么我们这里默认是函数级别的,后面会具体说fixture的作用范围)

实例

# test_fixture.py

import pytest

@pytest.fixture(autouse=True)

def fixtureFunc():

print('\n fixture->fixtureFunc')

def test_fixture():

print('in test_fixture')

class TestFixture(object):

def test_fixture_class(self):

print('in class with text_fixture_class')

if __name__=='__main__':

pytest.main(['-v', 'test_fixture.py'])

结果

fixture->fixtureFunc

.in test_fixture

fixture->fixtureFunc

.in class with text_fixture_class

[100%]

========================== 2 passed in 0.04 seconds ===========================

从结果可以看到每个测试用例执行前都自动执行了fixture

小结

掌握上面的方法,就可以使用fixture了,那么这几种方式又有是区别呢? 其实从我写的代码中就能看出来, 如果测试用例需要使用fixture中返回的参数,那么通过后面这两种方式是无法使用返回的参数的,因为fixture中返回的数据默认存在fixture名字里面存储,所以只能使用第一种方式才可以调用fixture中的返回值。(理论永远是理论,看文章的老铁还是自己试试吧!)

fixtur作用范围上面所有的实例默认都是函数级别的,所以测试函数只要调用了fixture,那么在测试函数执行前都会先指定fixture。说到作用范围就不得不说fixture 的第二个参数scope参数。

scope参数可以是session, module,class,function; 默认为function

1.session 会话级别(通常这个级别会结合conftest.py文件使用,所以后面说到conftest.py文件的时候再说)

2.module 模块级别: 模块里所有的用例执行前执行一次module级别的fixture

3.class 类级别 :每个类执行前都会执行一次class级别的fixture

4.function :前面实例已经说了,这个默认是默认的模式,函数级别的,每个测试用例执行前都会执行一次function级别的fixture

下面我们通过一个实例具体看一下 fixture的作用范围

# test_fixture.py

import pytest

@pytest.fixture(scope='module', autouse=True)

def module_fixture():

print('\n-----------------')

print('我是module fixture')

print('-----------------')

@pytest.fixture(scope='class')

def class_fixture():

print('\n-----------------')

print('我是class fixture')

print('-------------------')

@pytest.fixture(scope='function', autouse=True)

def func_fixture():

print('\n-----------------')

print('我是function fixture')

print('-------------------')

def test_1():

print('\n 我是test1')

@pytest.mark.usefixtures('class_fixture')

class TestFixture1(object):

def test_2(self):

print('\n我是class1里面的test2')

def test_3(self):

print('\n我是class1里面的test3')

@pytest.mark.usefixtures('class_fixture')

class TestFixture2(object):

def test_4(self):

print('\n我是class2里面的test4')

def test_5(self):

print('\n我是class2里面的test5')

if __name__=='__main__':

pytest.main(['-v', '--setup-show', 'test_fixture.py'])

运行结果

我们在cdm里面执行使用 --setup-show 可以查看到具体setup和teardoen顺序

test_fixture.py

SETUP M module_fixture

SETUP F func_fixture

-----------------

我是module fixture

-----------------

-----------------

我是function fixture

-------------------

test_fixture.py::test_1 (fixtures used: func_fixture, module_fixture).

我是test1

TEARDOWN F func_fixture

SETUP C class_fixture

SETUP F func_fixture

-----------------

我是class fixture

-------------------

-----------------

我是function fixture

-------------------

test_fixture.py::TestFixture1::test_2 (fixtures used: class_fixture, func_fixture, module_fixture).

我是class1里面的test2

TEARDOWN F func_fixture

SETUP F func_fixture

-----------------

我是function fixture

-------------------

test_fixture.py::TestFixture1::test_3 (fixtures used: class_fixture, func_fixture, module_fixture).

我是class1里面的test3

TEARDOWN F func_fixture

TEARDOWN C class_fixture

SETUP C class_fixture

SETUP F func_fixture

-----------------

我是class fixture

-------------------

-----------------

我是function fixture

-------------------

test_fixture.py::TestFixture2::test_4 (fixtures used: class_fixture, func_fixture, module_fixture).

我是class2里面的test4

TEARDOWN F func_fixture

SETUP F func_fixture

-----------------

我是function fixture

-------------------

test_fixture.py::TestFixture2::test_5 (fixtures used: class_fixture, func_fixture, module_fixture).

我是class2里面的test5

TEARDOWN F func_fixture

TEARDOWN C class_fixture

TEARDOWN M module_fixture

========================== 5 passed in 0.05 seconds ===========================

我们可以很清楚的看到 整个模块只执行了一次module级别的fixture , 每个类分别执行了一次class级别的fixture, 而每一个函数之前都执行了一次function级别的fixture

fixture实现teardown

其实前面的所有实例都只是做了测试用例执行之前的准备工作,那么用例执行之后该如何实现环境的清理工作呢?这不得不说yield关键字了,相比大家都或多或少的知道这个关键字,他的作用其实和return差不多,也能够返回数据给调用者,唯一的不同是被掉函数执行遇到yield会停止执行,接着执行调用处的函数,调用出的函数执行完后会继续执行yield关键后面的代码(具体原理可以看下我之前的文章关于生成器)。看下下面的实例来了解一下如何实现teardown功能

import pytest

from selenium import webdriver

import time

@pytest.fixture()

def fixtureFunc():

'''实现浏览器的打开和关闭'''

driver = webdriver.Firefox()

yield driver

driver.quit()

def test_search(fixtureFunc):

'''访问百度首页,搜索pytest字符串是否在页面源码中'''

driver = fixtureFunc

driver.get('http://www.baidu.com')

driver.find_element_by_id('kw').send_keys('pytest')

driver.find_element_by_id('su').click()

time.sleep(3)

source = driver.page_source

assert 'pytest' in source

if __name__=='__main__':

pytest.main(['--setup-show', 'test_fixture.py'])

这个实例会先打开浏览器,然后执行测试用例,最后关闭浏览器。大家可以试试! 通过yield就实现了 用例执行后的teardown功能

总结

1.fixture如何定义

2.fixture的使用方式

3.fixture作用范围

4.fixture用yield实现teardown功能

最后提一句:实际工作中尽量少用auto=True这个参数,可能会引发意想不到的结果! 最常用的还是通过传递参数最好!

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

python fixture_python pytest进阶之fixture详解相关推荐

  1. python fixture_python pytest进阶之xunit fixture详解

    前言 今天我们再说一下pytest框架和unittest框架相同的fixture的使用, 了解unittest的同学应该知道我们在初始化环境和销毁工作时,unittest使用的是setUp,tearD ...

  2. Pytest框架之fixture详解

    1.简介: fixture区别于unnitest的传统单元测试(setup/teardown)有显著改进: 1.有独立的命名,并通过声明它们从测试函数.模块.类或整个项目中的使用来激活. 2.按模块化 ...

  3. python 拼音库_python有没有拼音库python进阶之socket详解

    Socket的英文原义是"孔"或"插座".作为BSD UNIX的进程通信机制,通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句 ...

  4. python镜像下载包_python包详解

    干货大礼包!21天带你轻松学Python(文末领取更多福利) 点击查看课程视频地址 本课程来自于千锋教育在阿里云开发者社区学习中心上线课程<Python入门2020最新大课>,主讲人姜伟. ...

  5. python中xlrd模块的使用详解

    python中xlrd模块的使用详解 两个问题: 什么是xlrd模块? 为什么使用xlrd模块? 1.什么是xlrd模块? python操作excel主要用到xlrd和xlwt这两个库,即xlrd是读 ...

  6. 攻防世界web进阶区Web_python_block_chain详解

    攻防世界web进阶区Web_python_block_chain详解 题目 详解 51% 双花攻击 题目 详解 我们整理一下 Announcement: The server has been res ...

  7. Python 标准库之 sys 模块详解

    Python sys 模块详解 1. 简介 "sys"即"system","系统"之意.该模块提供了一些接口,用于访问 Python 解释器 ...

  8. 21年最新Python面试题及答案汇总详解(上)

    错过三月找工作的机会,还要错过四月的好时期吗?Python面试你做准备了吗?下面小编整理了一套2021年最新Python常见面试题目,及Python面试题目答案汇总.希望能够帮助到大家. 21年最新P ...

  9. python popen阻塞_对Python subprocess.Popen子进程管道阻塞详解

    问题产生描述 使用子进程处理一个大的日志文件,并对文件进行分析查询,需要等待子进程执行的输出结果,进行下一步处理. 出问题的代码 # 启用子进程执行外部shell命令 def __subprocess ...

最新文章

  1. 直接上手!不容错过的Visual Studio Code十大扩展组件
  2. 使用RNNs进行机器翻译——介绍RNN和LSTM网络及其应用
  3. [应用][js+css3]3D盒子导航[PC端]
  4. control theory and application
  5. python中time库引用不正确的_python模块知识一 自定义模块、time、datetime时间模块...
  6. arduino lora通讯_ESP8266 LoRa网关第一步:SPI总线
  7. java.lang.OutOfMemoryError: Java heap space异常
  8. Tomcat下载及安装(Windows)
  9. 51单片机LCD1602显示
  10. linux服务器架设篇 下载_后渗透系列——下载(Windows篇)
  11. sencha touch总结
  12. [微软拼音小技巧] 如何用Unicode输入生僻字
  13. IDEA设置终端Terminal窗口字体和大小
  14. Android 自定义下拉菜单的实现(基于PopupWindow+RecyclerView)
  15. Golang淘宝开放平台Api请求基础SDK
  16. amd 服务器芯片组,锐龙好搭档 推荐几款AMD芯片组主板
  17. 商睿智能科技完成千万元级别天使轮融资 1
  18. Android 蓝牙耳机 语音输入与播放
  19. R语言学习笔记(十四):聚类分析
  20. win任务栏计算机变未知,如何解决Win7打开程序出现异常未知的软件异常

热门文章

  1. python实现求1!+...+10!之和
  2. pods处于ContainerCreating状态
  3. 【图像处理】-019 补色
  4. 奔三的年纪,需要知道的44件事
  5. 如何选购自己满意的主板(建议观看)
  6. 华为OD机试真题-查找充电设备组合【2023Q1】【JAVA、Python、C++】
  7. Houdini SHOP学习笔记3 制作简单山体模型并制作简单材质
  8. 锦到说VBA——汇总一个文件夹下面的所有文件到EXCEL中
  9. ArcGIS Pro地图和场景
  10. 我的世界服务器怎么修改领地提示音,怎样设置领地权限 领地各项设置指令详解...