百科狗-知识改变命运!
--

Python中的Mock

是丫丫呀1年前 (2023-11-21)阅读数 10#技术干货
文章标签方法

1.前言

微服务架构下,由于各类服务开发进度的不一致,导致联调工作经常会存在不确定性,进而导致项目延期

在实际工作中,为了保证项目进度,我们经常需要针对部分未完成模块及不稳定模块采用Mock方式,以验证已开发完的模块

本篇文章将介绍Python实现Mock的几种常见方式

2.Mock介绍

Mock测试:在测试验证过程中,对于那些尚未完成或不稳定的对象,用一个虚拟对象来替代,以便测试的测试方法

因此,这个虚拟的对象是Mock对象,Mock对象是真实对象在调试期间的代替品

它的优势包含:

·前、后端并行开发

·模拟无法访问的资源

·隔离系统,避免脏数据干扰测试结果

3.1mock

在Python3.3之前使用mock,需要先安装依赖

#安装mock依赖

pip3installmock

项目地址:

https://github.com/testing-cabal/mock

假设Product类中有2个方法

·get_product_status_by_id

·buy_product

其中,get_product_status_by_id方法还没有实现;buy_product方法依赖于get_product_status_by_id方法的返回值。

#product_impl.py

classProduct(object):

def__init__(self):

pass

defget_product_status_by_id(self,product_id):

"""

Python中的Mock

通过商品id获取产品信息(Mock)

:return:

"""

#待实现查询数据库的业务逻辑

pass

defbuy_product(self,product_id):

"""

购买产品(真实逻辑)

:return:

"""

#产品信息

#{"id":1,"name":"苹果","num":23}

product=self.get_product_status_by_id(product_id)

ifproduct.get("num")>=1:

result={"status":0,"msg":"购买成功!"}

else:

result={"status":1,"msg":"购买失败,库存不足!"}

returnresult

Mock的步骤如下:

·导入使用mock中的patch方法

·作为测试方法的装饰器,对get_product_status_by_id方法进行Mock,方法参数为Mock对象

·测试方法中,对该Mock对象设置一个返回值

·调用并断言

frommockimportpatch

frommock_.product_implimportProduct

@patch('mock_.product_impl.Product.get_product_status_by_id')

deftest_succuse(mock_get_product_status_by_id):

#Mock方法,指定一个返回值

mock_get_product_status_by_id.return_value={"id":1,"name":"苹果","num":23}

product=Product()

assertproduct.buy_product(1).get("status")==0

需要注意的是,Mock此方法的时候,必须制定该方法的完整路径

使用@patch.object同样能完成Mock,不同的是,@patch.object包含2个参数

第一个参数为该方法所在的类;第二个参数为方法名

frommockimportpatch

frommock_.product_implimportProduct

#Mock一个方法

#@patch.object:对象、方法名

@patch.object(Product,'get_product_status_by_id')

deftest_succuse(mock_get_product_status_by_id):

#Mock方法,指定一个返回值

mock_get_product_status_by_id.return_value={"id":1,"name":"苹果","num":23}

product=Product()

assertproduct.buy_product(1).get("status")==0

3.2unittest.mock

Python3.3之后,mock作为标准库,已经内置到unittest中了

还是以3.1的场景为例,使用unittest编写一个测试用例

Mock步骤如下:

·导入unittest框架中的mock文件

·实例化Product对象

·mock.Mock(return_value=*)方法

·对get_product_status_by_id方法进行Mock

·调用并断言

importunittest

fromunittestimportmock

fromunittest_mock.product_implimportProduct

classTestProduct(unittest.TestCase):

deftest_success(self):

#成功结果

mock_success_value={"id":1,"name":"苹果","num":23}

product=Product()

product.get_product_status_by_id=mock.Mock(return_value=mock_success_value)

#调用实际函数

assertproduct.buy_product(1).get("status")==0

if__name__=="__main__":

unittest.main()

3.3pytest.mock

相比unittest,pytest由于强大的插件支持,用户群体可能更大!

如果项目本身使用的框架是pytest,则Mock更建议使用pytest-mock这个插件

#pytest依赖

pip3installpytest

Mock步骤如下:

·使用pytest编写测试方法,参数为mocker

·实例化Product对象

·使用mocker.patch()方法对get_product_status_by_id方法进行Mock,并设置返回值

·调用并断言

importpytest

frompytest_mock_.product_implimportProduct

deftest_buy_product_success(mocker):

"""

购买成功Mock

:parammocker:

:return:

"""

#实例化一个产品对象

product=Product()

#对Product中的方法的返回值进行Mock

mock_value={"id":1,"name":"苹果","num":23}

#Mock方法

#注意:需要指定方法的完整路径

#mocker.patch的第一个参数必须是模拟对象的具体路径,第二个参数用来指定返回值

product.get_product_status_by_id=mocker.patch("product_impl.Product.get_product_status_by_id",

return_value=mock_value)

#调用购买产品的方法

result=product.buy_product(1)

assertresult.get("status")==0

需要注意的是,mocker.patch方法第一个参数必须是Mock对象的完整路径

4.最后

文中对Python中常见的Mock方案进行了讲解,实际应用中,建议根据项目实际情况进行选型

以上内容为大家介绍了Python中的Mock,希望对大家有所帮助,如果想要了解更多Python相关知识,请关注IT培训机构:开发教育。http://www.baikegou.com/

鹏仔微信 15129739599 鹏仔QQ344225443 鹏仔前端 pjxi.com 共享博客 sharedbk.com

免责声明:我们致力于保护作者版权,注重分享,当前被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理!邮箱:344225443@qq.com)

图片声明:本站部分配图来自网络。本站只作为美观性配图使用,无任何非法侵犯第三方意图,一切解释权归图片著作权方,本站不承担任何责任。如有恶意碰瓷者,必当奉陪到底严惩不贷!

内容声明:本文中引用的各种信息及资料(包括但不限于文字、数据、图表及超链接等)均来源于该信息及资料的相关主体(包括但不限于公司、媒体、协会等机构)的官方网站或公开发表的信息。部分内容参考包括:(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供参考使用,不准确地方联系删除处理!本站为非盈利性质站点,本着为中国教育事业出一份力,发布内容不收取任何费用也不接任何广告!)