• 0
  • 0
分享
  • 单元测试之mock使用——软件测试圈
  • quinn 2022-11-21 10:54:52 字数 2829 阅读 2411 收藏 0

一、简介

 一般程序中A类的m1方法调用B类的m2方法,而B类的m2方法又调用了C类的m3方法以此类推等等,而其中的某个方法的一些数据又需要调用其它服务或者查询数据库,一般单元测试只针对某个功能进行测试,但是如上面的情况在做单元测试时受程序结构、环境等条件限制就会变得非常复杂。mock可以模拟对象返回方式来解决与该单元功能不相关的依赖关系,即模拟B类的m2方法返回结果来进行A类的m1方法单元测试,排除受到B类C类等其它不相关因素的影响。

二、使用mock做单元测试的优点

1、效率高

就是跑Java代码,不需要启用Spring及连接数据库。

2、TDD(测试驱动开发)

即先编写单元测试用例,根据单元测试用例再编写程序代码。

3、并行开发

团队有多个开发人员时在把数据格式约定好后可使用mock模拟返回结果而不需等待功能开发完成后才能联调。

4、解决环境依赖问题

比如当网络不通、服务无法访问时也能进行单元测试。

三、mock实现原理

使用Stub(桩)技术动态的替换原程序功能。即在程序运行时将原本程序要调用class字节码的逻辑替换为mock代码。

四、mock使用

1、导入Maven依赖包

<dependency>
    <groupId>org.mockito</groupId>
    <artifactId>mockito-core</artifactId>
    <version>2.8.9</version>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>org.powermock</groupId>
    <artifactId>powermock-module-junit4</artifactId>
    <version>1.7.4</version>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>org.powermock</groupId>
    <artifactId>powermock-api-mockito2</artifactId>
    <version>1.7.4</version>
    <scope>test</scope>
</dependency>

2、编写测试代码

2.1、模块接口调用代码

1.png

2.2、模块业务处理代码

2.png

 2.3、mock单元测试代码

/**
 * create with Daniel
 * Date: 2022/4/12
 **/
@RunWith(MockitoJUnitRunner.class)
public class MockTest {
    //注入业务对象
    @InjectMocks
    private Service service;
    //使用Mock对象
    @Mock
    private FeignService feignService;
 
    /**
    * 测试方法
    * @author: Daniel
    * @date: 2022/4/12
    * @return void
    */
    @Test
    public void tester(){
        //初始化动作
        MockitoAnnotations.initMocks(this);
        Mockito.when(feignService.doApi(Mockito.anyString())).thenReturn("我是Mock对象");
        String result = service.doSomething("name");
        System.out.println(result);
    }
}

运行结果

3.png

从上图可以发现在Service调用Feign方法时返回的不是Feign的功能代码逻辑,而是返回测试用例中Mock返回的代码,这样可有效的解决单元测试用例对环境依赖的问题。

2.4、mock的其它用法

        //方法多次调用返回不同的值
        Mockito.when(feignService.doApi(Mockito.anyString())).thenReturn("第一次调用").thenReturn("第二次调用");
        System.out.println(service.doSomething("name"));
        System.out.println(service.doSomething("name"));
        //Mock没有返回值方法
        Mockito.doNothing().when(feignService).noReturn();
        //模块异常情况
        Mockito.when(feignService.doApi(Mockito.anyString())).thenThrow(new RuntimeException("我是Mock异常"));
        try{
            service.doSomething("name");
        }catch (Exception e){
            System.out.println(e.getMessage());
        }

运行结果

4.png

3、PowerMock使用 

PowerMock主要用于模拟静态方法的调用。

3.1、增加工具测试类

5.png

3.2、Mock测试类调整

 在类上加以下二个注解

@RunWith(PowerMockRunner.class)
@PrepareForTest({MockUtils.class})

添加测试代码

@Test
    public void testStatic(){
        PowerMockito.mockStatic(MockUtils.class);
        PowerMockito.when(MockUtils.check(Mockito.any())).thenReturn(true);
        System.out.println(MockUtils.check(null));
        System.out.println(MockUtils.check(1));
        System.out.println(MockUtils.check(new Object()));
    }

运行结果

6.png

由些可见,无论传什么参数都永远返回True。 


作者:不甘于平凡的溃败

原文链接:https://blog.csdn.net/wohiusdashi/article/details/124085245

  • 【留下美好印记】
    赞赏支持
登录 后发表评论
+ 关注

热门文章

    最新讲堂

      • 推荐阅读
      • 换一换
          • 开发在开发项目时,并不是所有功能都从头开始写的,相同的功能是调用公共组件或者统一方法,这样写出来可以保证产品风格统一,对于一些特殊场景处理一致,即使出现问题,只需要改一遍就可以,不需要每个地方都修改,省时省力,方便维护。与开发公共组件相对应的,测试有没有公共用例呢?答案是肯定的,公共用例分两种,功能公共用例,规则公共用例。功能公共用例是与当前系统功能紧密挂钩的,且是经常出错的地方或者用户在意的地方,跟行业是相关的,只要调用这块功能接口的都需要慎重。规则公共用例是放之四海而皆准的,只要是涉及到这种类型测试的都可以复用,跟行业关系不大,有助于测试思路扩展。多多提取公共用例,对于自己测试经验的积累,...
            6 4 8384
            分享
          • 国内软件业快速发展的最近十年,软件开发工程师的人数和职业水平得到了很大的提高,当前国内高水平的软件开发工程师的数量已经可以和许多软件业发达的国家相比。但是,软件测试人才严重缺乏,尤其是既懂质量管理,又懂测试技术的软件测试工程师,更是凤毛麟角。现阶段软件测试工程师的晋升通道有两种:一种是专业通道,成长为高级软件测试工程师或专职的性能测试工程师、自动化测试工程师、白盒测试工程师,这时能够独立测试很多软件,甚至可以成为软件测试架构设计师,当然随着技术的积累也可以转做项目管理;第二种是管理通道,从测试工程师到组长(Lead),再到测试经理(Manager),以至更高的职位。测试工程师晋升通道如图3-6...
            0 0 1058
            分享
          • Cypress与Selenium/WebDriverSelenium/WebDriver架构它基于Client/Server架构设计,其架构图如下所示Language Bindings/Client也叫做ClientLibrary,它是Selenium框架的一系列jar文件,可以使用不同的编程语言编写,也正因为它的存在,才使得各个语言编写的测试代码能够被正确解析浏览器驱动WebDriverWebDriver用于管理和完全控制浏览器,根据不同的浏览器区分不同的DriverHTTP传输的JSON Wire协议JSON(JavaScript Object Notation)是一种在Web上的服务器端...
            13 13 2406
            分享
          •   测试小伙伴们,你们有遇到下图的情况吗?  这张图其实还算“温柔”的,其实有些情况下,某些测试人员或者开发人员脾气大的可能撕逼或者快干架。所以如何和开发有效沟通,并高效劝说开发改掉bug是一门学问,以下是我总结八年测试经验给测试新人的一些建议:  1、和开发人员保持友好的团队关系。这是最重要的一点!  我以前遇到一个开发,刚开始给他提bug时,他是各种抵触情绪加敷衍。后来我就私底下和他多接触,了解他的脾气,久而久之他也和我熟络起来,结果不仅不再有抵触情绪,甚至还帮我主动定位bug。其实人心都是肉长的,我们做事既要讲理,也要适当打打“感情牌”。注意跟开发沟通的语气,要有换位思考的意识,做事情对...
            1 1 1550
            分享
          •   本文以笔者当前使用的自动化测试项目为例,浅谈分层设计的思路,不涉及到具体的代码细节和某个框架的实现原理,重点关注在分层前后的使用对比,可能会以一些伪代码为例来说明举例。  接口测试三要素:  ·参数构造  · 发起请求,获取响应  · 校验结果  一、原始状态  当我们的用例没有进行分层设计的时候,只能算是一个“苗条式”的脚本。以一个后台创建商品活动的场景为例,大概流程是这样的(默认已经是登录状态下):  创建商品-创建分类-创建优惠券-创建活动  要进行接口测试的话,按照接口测试的三要素来进行,具体的效果如下:  参数构造:     &n...
            0 0 1137
            分享
      • 51testing软件测试圈微信