Spring依赖注入与mock

作者: 武可 | 来源:发表于2018-09-09 11:43 被阅读159次

一般使用Spring,都会用到依赖注入(DI)。

@Service
public class SampleService {
    @Autowired
    private SampleDependency dependency;
    public String foo() {
        return dependency.getExternalValue("bar");
    }
}

如果测试中需要对Sping注入的对象进行注入,该怎么做呢?

选择一 修改实现

一种做法是把字段注入改为构造函数注入:

@Service
public class SampleService {
    private SampleDependency dependency;
    @Autowired
    public SampleService(SampleDependency dependency, PersonPoolProvider personPoolProvider) {
        this.dependency = dependency;
    }
}

或者属性注入:

private SampleDependency dependency;
@Autowired
public void setDependency(SampleDependency dependency) {
    this.dependency = dependency;
}

测试就可以写成

SampleDependency dependency = mock(SampleDependency.class);
SampleService service = new SampleService(dependency);

从道理来讲这样更加规范一些。不过事实上会产生更多的代码,在字段增删的时候、构造函数、getter也需要随之维护。

选择二 绕过限制

也可以用一些绕过访问级别的“黑魔法”,比如测试写成这样

SampleDependency dependency = mock(SampleDependency.class);
SampleService service = new SampleService();
ReflectionTestUtils.setField(service, "dependency", dependency);

总感觉不太优雅,而且万一字段改名也很可能漏改。

当然,也可以直接把字段改为package可见甚至public。不过总觉得对不起自己的代码洁癖。

选择三 使用Mockito InjectMocks

这里推荐使用mockito 的InjectMocks注解。测试可以写成

@Rule public MockitoRule rule = MockitoJUnit.rule();
@Mock SampleDependency dependency;
@InjectMocks SampleService sampleService;

对应于实现代码中的每个@Autowired字段,测试中可以用一个@Mock声明mock对象,并用@InjectMocks标示需要注入的对象。

这里的MockitoRule的作用是初始化mock对象和进行注入的。有三种方式做这件事。

  • 测试@RunWith(MockitoJUnitRunner.class)
  • 使用rule @Rule public MockitoRule rule = MockitoJUnit.rule();
  • 调用 MockitoAnnotations.initMocks(this),一般在setup方法中调用

InjectMocks可以和Sping的依赖注入结合使用。比如:

@RunWith(SpringRunner.class)
@DirtiesContext
@SpringBootTest
public class ServiceWithMockTest {
    @Rule public MockitoRule rule = MockitoJUnit.rule();
    @Mock DependencyA dependencyA;
    @Autowired @InjectMocks SampleService sampleService;

    @Test
    public void testDependency() {
        when(dependencyA.getExternalValue(anyString())).thenReturn("mock val: A");
        assertEquals("mock val: A", sampleService.foo());
    }
}

假定Service注入了2个依赖dependencyA, dependencyB。上面测试使用Spring注入了B,把A替换为mock对象。

需要注意的是,Spring test默认会重用bean。如果另有一个测试也使用注入的SampleService并在这个测试之后运行,那么拿到service中的dependencyA仍然是mock对象。一般这是不期望的。所以需要用@DirtiesContext修饰上面的测试避免这个问题。

选择四 Springboot MockBean

如果使用的是Springboot,测试可以用MockBean更简单的写出等价的测试。

@RunWith(SpringRunner.class)
@SpringBootTest
public class ServiceWithMockBeanTest {
    @MockBean SampleDependencyA dependencyA;
    @Autowired SampleService sampleService;

    @Test
    public void testDependency() {
        when(dependencyA.getExternalValue(anyString())).thenReturn("mock val: A");
        assertEquals("mock val: A", sampleService.foo());
    }
}

相关文章

  • Spring依赖注入与mock

    一般使用Spring,都会用到依赖注入(DI)。 如果测试中需要对Sping注入的对象进行注入,该怎么做呢? 选择...

  • .net core 3.1 单元测试

    直接上代码 借助Mock来解决依赖注入的问题 mock+实际依赖(mapper)直接Mock new 对象的方式...

  • spring

    spring4介绍 spring配置详解 IOC(控制反转) 与 DI(依赖注入)重点 依赖注入各种类型const...

  • Spring学习手册(5)—— bean作用域

    Spring学习手册(4)—— Spring 依赖注入中介绍了Spring依赖注入的基本方式以及各种类型的参数注入...

  • mock.js使用

    1.引入mock.js依赖 package.jsdependencies依赖注入mockjs ("mockjs":...

  • spring之控制反转和依赖注入

    Spring学习(三)IOC控制反转与DI依赖注入

  • Spring学习之依赖注入

    Spring学习之依赖注入 依赖注入的基本概念 依赖注入(Dependecy Injection),也称为IoC(...

  • Spring 框架结构

    Spring 中文手册. Core spring-core:依赖注入IoC与DI的最基本实现spring-bean...

  • 2018-05-05

    spring源码分析(三) 目录五、Spring 源码解读--5.4、IOC 容器的依赖注入----1、依赖注入发...

  • Spring MVC使用

    1、spring 与spring MVC的不是一个东西,spring依赖注入,aop,spring全家桶2、web...

网友评论

    本文标题:Spring依赖注入与mock

    本文链接:https://www.haomeiwen.com/subject/hqqhgftx.html