百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

Python 中 Mock 到底该怎么玩?一篇文章告诉你

off999 2024-12-27 15:59 14 浏览 0 评论

来源:AirPython

作者:星安果

1. 前言

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

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

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

2. Mock 介绍

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

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

它的优势包含:

  • 前、后端并行开发
  • 模拟无法访问的资源
  • 隔离系统,避免脏数据干扰测试结果
  • 3.1 mock

    在 Python 3.3 之前使用 mock,需要先安装依赖

    # 安装mock依赖
    pip3 install 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
    class Product(object):
    def __init__(self):
    pass
    def get_product_status_by_id(self, product_id):
    """
    通过商品id获取产品信息(Mock)
    :return:
    """

    # 待实现查询数据库的业务逻辑
    pass
    def buy_product(self, product_id):
    """
    购买产品(真实逻辑)
    :return:
    """

    # 产品信息
    # {"id":1,"name":"苹果","num":23}
    product = self.get_product_status_by_id(product_id)
    if product.get("num") >= 1:
    result = {
    "status": 0, "msg": "购买成功!"}
    else:
    result = {
    "status": 1, "msg": "购买失败,库存不足!"}
    return result

    Mock 的步骤如下:
    导入使用 mock 中的 patch 方法作为测试方法的装饰器,对 get_product_status_by_id 方法进行 Mock,方法参数为 Mock 对象测试方法中,对该 Mock 对象设置一个返回值调用并断言from mock import patch
    from mock_.product_impl import Product
    
    @patch('mock_.product_impl.Product.get_product_status_by_id')
    def test_succuse(mock_get_product_status_by_id):
        # Mock方法,指定一个返回值
        mock_get_product_status_by_id.return_value = {"id": 1, "name": "苹果", "num": 23}
    
        product = Product()
    
        assert product.buy_product(1).get("status") == 0
    
    需要注意的是,Mock 此方法的时候,必须制定该方法的完整路径使用 @patch.object 同样能完成 Mock,不同的是,@patch.object 包含 2 个参数第一个参数为该方法所在的类;第二个参数为方法名from mock import patch
    
    from mock_.product_impl import Product
    
    # Mock一个方法
    # @patch.object:对象、方法名
    @patch.object(Product, 'get_product_status_by_id')
    def test_succuse(mock_get_product_status_by_id):
        # Mock方法,指定一个返回值
        mock_get_product_status_by_id.return_value = {"id": 1, "name": "苹果", "num": 23}
    
        product = Product()
    
        assert product.buy_product(1).get("status") == 0

    3.2 unittest.mock

    Python 3.3 之后,mock 作为标准库,已经内置到 unittest 中了

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

    Mock 步骤如下:

  • 导入 unittest 框架中的 mock 文件
  • 实例化 Product 对象
  • mock.Mock(return_value=*) 方法对 get_product_status_by_id 方法进行 Mock
  • 调用并断言
  • import unittest
    from unittest import mock
    from unittest_mock.product_impl import Product
    class TestProduct(unittest.TestCase):
    def test_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)
    # 调用实际函数
    assert product.buy_product(1).get("status") == 0
    if __name__ == "__main__":
    unittest.main()

    3.3 pytest.mock

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

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

    # pytest依赖
    pip3 install pytest

    Mock 步骤如下:

  • 使用 pytest 编写测试方法,参数为 mocker
  • 实例化 Product 对象
  • 使用 mocker.patch() 方法对 get_product_status_by_id 方法进行 Mock,并设置返回值
  • 调用并断言
  • import pytest
    from pytest_mock_.product_impl import Product
    def test_buy_product_success(mocker):
    """
    购买成功Mock
    :param mocker:
    :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)
    assert result.get("status") == 0

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

    4. 最后

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

    如果你觉得文章还不错,请大家 点赞、分享、留言 下,因为这将是我持续输出更多优质文章的最强动力!

    相关推荐

    独家 | 5 个Python高级特性让你在不知不觉中成为Python高手

    你已经使用Python编程了一段时间,编写脚本并解决各种问题。是你的水平出色吗?你可能只是在不知不觉中利用了Python的高级特性。从闭包(closure)到上下文管理器(contextmana...

    Python装饰器

    Python装饰器是一种用于修改函数或类的行为的特殊语法。它们允许在不修改原始代码的情况下,通过将函数或类作为参数传递给另一个函数来添加额外的功能。装饰器本质上是一个函数,它接受一个函数作为参数,并返...

    中高阶Python常规用法--上下文管理器

    Python以简单性和通用性著称,是一种深受全球开发人员喜爱的编程语言。它提供了大量的特性和功能,使编码成为一种愉快的体验。在这些功能中,一个经常被新手忽视的强大工具是上下文管理器。上下文管理器是高...

    Python小案例67- 装饰器

    Python装饰器是一种用于修改函数或类的行为的特殊语法。它们允许在不修改原始代码的情况下,通过将函数或类作为参数传递给另一个函数来添加额外的功能。装饰器本质上是一个函数,它接受一个函数作为参数,并返...

    python常用的语法糖

    概念Python的语法糖(SyntacticSugar)是指那些让代码更简洁、更易读的语法特性,它们本质上并不会增加新功能,但能让开发者更高效地编写代码。推导式写法推导式是Python最经典的...

    python - 常用的装饰器 decorator 有哪些?

    python编程中使用装饰器(decorator)工具,可以使代码更简洁清晰,提高代码的重用性,还可以为代码维护提供方便。对于python初学者来说,根据装饰器(decorator)的字面意思并不...

    python数据缓存怎么搞 ?推荐一个三方包供你参考,非常简单好用。

    1.数据缓存说明数据缓存可以说也是项目开发中比不可少的一个工具,像我们测试的系统中,你都会见到像Redis一样的数据缓存库。使用缓存数据库的好处不言而喻,那就是效率高,简单数据直接放在缓存中...

    用于时间序列数据的Graphite监视工具

    结合第三方工具,Graphite为IT性能监控提供了许多好处。本文介绍其核心组件,包括Carbon、Whisper以及安装的基本准则。Graphite监视工具可实时或按需,大规模地绘制来自多个来源的时...

    Python3+pygame实现的坦克大战

    一、显示效果二、代码1.说明几乎所有pygame游戏,基本都遵循一定的开发流程,大体如下:初始化pygame创建窗口while循环检测以及处理事件(鼠标点击、按键等)更新UI界面2.代码创建一个m...

    Python之鸭子类型:一次搞懂with与上下文装饰器

    引言在鸭子类型的理念的基础之上,从关注类型,转变到关注特性和行为。结合Python中的魔法函数的体系,我们可以将自定义的类型,像内置类型一样被使用。今天这篇文章中,接着该话题,继续聊一下with语法块...

    Python必会的50个代码操作

    学习Python时,掌握一些常用的程序操作非常重要。以下是50个Python必会的程序操作,主要包括基础语法、数据结构、函数和文件操作等。1.HelloWorldprint("Hello,...

    一文掌握Python 中的同步和异步

    同步代码(Sync)同步就像在一个流水线上工作,每个任务都等待前一个任务完成。示例:机器A切割钢板→完成后,机器B钻孔→完成后,机器C上色。在Python中,同步代码看起来像这样:im...

    python 标注模块timeit: 测试函数的运行时间

    在Python中,可以使用内置的timeit模块来测试函数的运行时间。timeit模块提供了一个简单的接口来测量小段代码的执行时间。以下是使用timeit测试函数运行时间的一般步骤:导入...

    Python带你找回童年的万花尺

    还记得小时候的万花尺吧?这么画:一点也不费脑筋,就可以出来这么多丰富多彩的复杂几何图形。具体而言,可以用万花尺玩具(如图2-1所示)来绘制数学曲线。这种玩具由两个不同尺寸的塑料齿轮组成,一大一小。小的...

    Python 时间模块深度解析:从基础到高级的全面指南

    直接上干货一、时间模块核心类介绍序号类名说明1datetime.datetime表示一个具体的日期和时间,结合了日期和时间的信息。2datetime.date表示一个具体的日期。3datetime.t...

    取消回复欢迎 发表评论: