首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Spring依赖注入与mock

Spring依赖注入与mock

作者头像
烟雨平生
发布2023-03-07 16:24:03
发布2023-03-07 16:24:03
88100
代码可运行
举报
文章被收录于专栏:数字化之路数字化之路
运行总次数:0
代码可运行
一般使用Spring,都会用到依赖注入(DI)。
代码语言:javascript
代码运行次数:0
运行
复制
@Service
public class SampleService {
    @Autowired
    private SampleDependency dependency;
    public String foo() {
        return dependency.getExternalValue("bar");
    }
}

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

选择一 修改实现

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

代码语言:javascript
代码运行次数:0
运行
复制
@Service
public class SampleService {
    private SampleDependency dependency;
    @Autowired
    public SampleService(SampleDependency dependency, PersonPoolProvider personPoolProvider) {
        this.dependency = dependency;
    }
}

或者属性注入:

代码语言:javascript
代码运行次数:0
运行
复制
private SampleDependency dependency;
@Autowired
public void setDependency(SampleDependency dependency) {
    this.dependency = dependency;
}

测试就可以写成

代码语言:javascript
代码运行次数:0
运行
复制
SampleDependency dependency = mock(SampleDependency.class);
SampleService service = new SampleService(dependency);

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

选择二 绕过限制

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

代码语言:javascript
代码运行次数:0
运行
复制
SampleDependency dependency = mock(SampleDependency.class);
SampleService service = new SampleService();
ReflectionTestUtils.setField(service, "dependency", dependency);

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

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

选择三 使用Mockito InjectMocks

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

代码语言:javascript
代码运行次数:0
运行
复制
@Rule public MockitoRule rule = MockitoJUnit.rule();
@Mock SampleDependency dependency;
@InjectMocks SampleService sampleService;

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

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

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

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

代码语言:javascript
代码运行次数:0
运行
复制
@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修饰上面的测试避免这个问题。

选择四 Spring Boot MockBean

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

代码语言:javascript
代码运行次数:0
运行
复制
@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());
    }
}
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2018-09-09,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 的数字化之路 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 选择一 修改实现
  • 选择二 绕过限制
  • 选择三 使用Mockito InjectMocks
  • 选择四 Spring Boot MockBean
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档