mockito入门
  0B0UEqYIF9j7 2023年11月02日 98 0

前言

最近在项目中跑单元测试发现直接使用springboot自带的测试,一整套跑起来花费数十分钟,这是无法忍受的,考虑到功能的特殊性,想到了Spring测试包自带的mockito单元测试,所以进行初次尝试使用。项目使用简单的一些类覆盖部分业务场景。

测试代码

pom包 image.png

<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-test</artifactId>
   <scope>test</scope>
</dependency>

<dependency>
   <groupId>org.mockito</groupId>
   <artifactId>mockito-inline</artifactId>
   <version>4.5.1</version>
   <scope>test</scope>
</dependency>

主要测试类

    import com.timelinecapital.web.service.order.*;
    import org.checkerframework.checker.units.qual.A;
    import org.junit.jupiter.api.Assertions;
    import org.junit.jupiter.api.Test;
    import org.junit.jupiter.api.extension.ExtendWith;
    import org.mockito.MockedStatic;
    import org.mockito.Mockito;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.boot.test.mock.mockito.MockBean;
    import org.springframework.test.context.ContextConfiguration;
    import org.springframework.test.context.junit.jupiter.SpringExtension;

    import static org.mockito.ArgumentMatchers.anyInt;
    import static org.mockito.BDDMockito.then;
    import static org.mockito.Mockito.when;

    /**
     * @author Steven
     * @Date 2023/1/30 15:48
     */
    @ContextConfiguration(classes = {OrderB.class, OrderA.class, OrderC.class, OrderD.class})
    @ExtendWith(SpringExtension.class)
    public class OrderTest {

        @Autowired
        private OrderB orderB;

        @MockBean
        private OrderC orderC;


        @Test
        public void testOrder() throws Exception {
            Mockito.mockStatic(OrderE.class);
    //        try (MockedStatic<OrderE> orderEMockedStatic = Mockito.mockStatic(OrderE.class)) {
                when(OrderE.print()).thenReturn(333);
    //            orderEMockedStatic.close();
    //        }
    //        when(orderA.print()).thenReturn(100);
    //        when(orderC.print(anyInt())).thenReturn(101);
            when(orderC.print(anyInt())).thenReturn(201);

            boolean res = orderB.test();

            Assertions.assertTrue(res);
        }

    }

测试代码说明:

@ExtendWith(SpringExtension.class) 这个注解重要是继承spring的环境

@ContextConfiguration 这个注解主要
    - classes属性主要是导入springbean,如果不配置需要倒入的bean会报依赖注入异常

默认情况需要手动配置相关需要注入bean的类否则会报错,依赖注入异常,找不到相应的bean

Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'orderB': Unsatisfied dependency expressed through field 'order'; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'com.timelinecapital.web.service.order.OrderA' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.resolveFieldValue(AutowiredAnnotationBeanPostProcessor.java:659)
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:639)
    at org.springframework.beans.factory.annotation.InjectionMetadata.inject(InjectionMetadata.java:119)

有些时间需要mock静态方法,单纯的使用Mockito.mockStatic(OrderE.class);会报错,需要导入mockito-inline包,具体原因后续分析常见的错误异常有,这错误异常很友好都提示了如何操作:

org.mockito.exceptions.base.MockitoException: 
The used MockMaker SubclassByteBuddyMockMaker does not support the creation of static mocks

Mockito's inline mock maker supports static mocks based on the Instrumentation API.
You can simply enable this mock mode, by placing the 'mockito-inline' artifact where you are currently using 'mockito-core'.
Note that Mockito's inline mock maker is not supported on Android.

总结

默认情况下需要手动配置相对应的所有需要注入的bean,凡是你这个模块需要倒入的bean你都需要配置,如果你不配置注入就会报错,要么注入,要么mock,比如OrderB里面依赖了OrderA和一个静态的OrderD类,OrderA bean要么mock要么注入,OrderD就不一样了因为是静态的,但是导入的时候需要注入OrderC所以站在Spring的角度,都需要注入。当然也可以一开始就mock,当一个实例被mock之后他就成虚拟的了,他的依赖就不需要进行注入了。 站在项目的角度,项目越大不可能为了某个功能跑一次服务,因为服务还有其他功能,所以功能测试就成了很好的解决方案,你可以随意的mock,返回想要的值,最大角度的覆盖所有测试,唯一的缺点是,你需要考虑所有的依赖注入。 静态类的或者实例的mock,一旦mock这个类的其他方法都会变成虚拟的,需要你配置场景才能返回对应的值,否则就像调用了一句空代码一样,好也好也不好,等我优化后续的代码上传一篇文章深入探讨实现。

【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月08日 0

暂无评论

推荐阅读
  YgmmQQ65rPv4   2023年11月19日   29   0   0 Java应用程序
  Y8XIq1u6ceQW   2023年11月19日   38   0   0 Java
  AeUHztwqqxTz   2023年11月02日   42   0   0 Javatomcatapache
0B0UEqYIF9j7
作者其他文章 更多
最新推荐 更多