最近在做某个项目的时候一直使用 @MockBean 来解决单元测试中 Mock 类装配到被测试类的问题。这篇文章主要介绍了 @MockBean 的使用例子以及不使用 @MockBean 而使用@SpyBean 的情景和原因。

文章中的所有代码均为 Kotlin 语言,与 Java 略有不同。但是 Kotlin 的语法比较容易理解,原生 Java 的读者在阅读时应该不会有太大的障碍。

请看下面的代码:

import java.net.URI

import org.springframework.beans.factory.annotation.Autowired
import org.springframework.http.HttpHeaders
import org.springframework.http.HttpStatus
import org.springframework.http.ResponseEntity
import org.springframework.web.bind.annotation.PostMapping
import org.springframework.web.bind.annotation.RequestMapping
import org.springframework.web.bind.annotation.RestController

@RestController
@RequestMapping(path = “/api/users”)
class UserResource {
@Autowired
private val userRepository:UserRepository

@PostMapping
fun create(request:CreateUserRequest):ResponseEntity {
val user = userRepository.save(request.toUser())
val headers = HttpHeaders()
headers.setLocation(URI.create("/api/users/" + user.getId()))
return ResponseEntity(headers, HttpStatus.CREATED)
}
}
这是一个简单的 Spring controller ,向外暴露了一个 /api/users 接口,并且注入了一个自定义的 repository( UserRepository )用来和 MongoDB 的数据库交流(限于篇幅,自定义的 repository 的实现代码没有给出)。

我们现在要对它做单元测试,下面是单元测试的代码:

import org.junit.Test
import org.junit.runner.RunWith
import org.mockito.Mockito

import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.boot.test.mock.mockito.MockBean
import org.springframework.http.MediaType
import org.springframework.test.context.junit4.SpringRunner
import org.springframework.test.web.servlet.MockMvc

import org.mockito.Mockito.when
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post
import org.springframework.test.web.servlet.result.MockMvcResultHandlers.print
import org.springframework.test.web.servlet.result.MockMvcResultMatchers.header
import org.springframework.test.web.servlet.result.MockMvcResultMatchers.status

@SpringBootTest
@RunWith(SpringRunner::class)
@AutoConfigureMockMvc
class UserResourceTests {
@Autowired
private val mockMvc:MockMvc

@MockBean
private val userRepository:UserRepository

@Test
fun should_create_a_user() {
val json = “{“username”:“shekhargulati”,“name”:“Shekhar Gulati”}”
when(userRepository.save(Mockito.any(User::class.java))).thenReturn(User(“123”))
this.mockMvc
.perform(post("/api/users").contentType(MediaType.APPLICATION_JSON).content(json))
.andDo(print())
.andExpect(status().isCreated())
.andExpect(header().string(“Location”, “/api/users/123”))
}
}
可以看到,在做单元测试时,如果想要 mock UserRepository 的逻辑,只需要声明一个变量并在上面加上 @MockBean 的注释即可,之后使用 when().thenReturn() 来设定 mock UserRepository 的行为。在运行时 SpringBoot 会扫描到你注释的 mock ,并自动装配到被测试的 controller 里面。这也是和 @Mock 注释不同的地方,后者只能生成一个 Mock 类,但是并不能自动装配到其它类里面。

MongoTemplate 的单元测试
现在假设你并没有使用自己实现的 UserRepository 来与数据库交流,而是使用 SpringBoot 自带的 MongoTemplate 装配到 controller 里面,那么代码大概是下面这样的:

@RestController
@RequestMapping(path = “/api/users”)
class UserResource {
@Autowired
private val mongoTemplate:MongoTemplate

@PostMapping
fun create(@RequestBody request:CreateUserRequest):ResponseEntity {
val user = this.mongoTemplate.findOne(
Query.query(Criteria.where(“username”).is(request.username)),
User::class.java
)
if (user != null)
{
return ResponseEntity(HttpStatus.CONFLICT)
}
mongoTemplate.save(request.toUser(), “user”)
return ResponseEntity(HttpStatus.CREATED)
}
}
可以看到代码的结构没有大的变化,只是不同的接口在方法调用的细节上不太一样。现在我们要对它做单元测试。

代码如下:

@SpringBootTest
@RunWith(SpringRunner::class)
@AutoConfigureMockMvc
class UserResourceTests {
@Autowired
private val mockMvc:MockMvc

@MockBean
private val mongoTemplate:MongoTemplate

@Test
fun should_create_a_user() {
val json = “{“username”:“shekhargulati”,“name”:“Shekhar Gulati”}”
when(mongoTemplate.findOne(Mockito.any(Query::class.java), Mockito.eq(User::class.java))).thenReturn(User(“123”))
this.mockMvc
.perform(post("/api/users").contentType(MediaType.APPLICATION_JSON).content(json))
.andDo(print())
.andExpect(status().isCreated())
verify(mongoTemplate).save(Mockito.any(User::class.java))
}
}
但是当运行的时候却出现了 NullPointerException。

Caused by: java.lang.NullPointerException
at org.springframework.data.mongodb.repository.support.MongoRepositoryFactory.(MongoRepositoryFactory.java:73)
at org.springframework.data.mongodb.repository.support.MongoRepositoryFactoryBean.getFactoryInstance(MongoRepositoryFactoryBean.java:104)
at org.springframework.data.mongodb.repository.support.MongoRepositoryFactoryBean.createRepositoryFactory(MongoRepositoryFactoryBean.java:88)
at org.springframework.data.repository.core.support.RepositoryFactoryBeanSupport.afterPropertiesSet(RepositoryFactoryBeanSupport.java:248)
at org.springframework.data.mongodb.repository.support.MongoRepositoryFactoryBean.afterPropertiesSet(MongoRepositoryFactoryBean.java:117)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1687)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1624)
之所以会出现空指针异常,是因为 MongoTemplate 是一个 SpringBoot 库的一个内部接口,而 @MockBean 只能 mock 本地的代码——或者说是自己写的代码,对于储存在库中而且又是以 Bean 的形式装配到代码中的类无能为力。

该 @SpyBean 上场了
@SpyBean 与 @Spy 的关系类似于 @MockBean 与 @Mock 的关系。和 @MockBean 不同的是,它不会生成一个 Bean 的替代品装配到类中,而是会监听一个真正的 Bean 中某些特定的方法,并在调用这些方法时给出指定的反馈。却不会影响这个 Bean 其它的功能。

于是测试代码变成了下面这样。

@SpringBootTest
@RunWith(SpringRunner::class)
@AutoConfigureMockMvc
class UserResourceTests {
@Autowired
private val mockMvc:MockMvc

@SpyBean
private val mongoTemplate:MongoTemplate

@Test
fun should_create_a_user() {
val json = “{“username”:“shekhargulati”,“name”:“Shekhar Gulati”}”
doReturn(null)
.when(mongoTemplate).findOne(Mockito.any(Query::class.java), Mockito.eq(User::class.java))
doNothing().when(mongoTemplate).save(Mockito.any(User::class.java))
this.mockMvc
.perform(post("/api/users").contentType(MediaType.APPLICATION_JSON).content(json))
.andDo(print())
.andExpect(status().isCreated())
verify(mongoTemplate).save(Mockito.any(User::class.java))
}
}
@SpyBean 包裹着真正的 Bean 装配到了 controller 中,并对特定的行为作出反应。

需要注意的是,设置 spy 逻辑时不能再使用 when(某对象.某方法).thenReturn(某对象) 的语法,而是需要使用 doReturn(某对象).when(某对象).某方法 或者 doNothing(某对象).when(某对象).某方法。

总结
@SpyBean 解决了 SpringBoot 的单元测试中 @MockBean 不能 mock 库中自动装配的 Bean 的局限。使 SpringBoot 的单元测试更灵活也更简单。

假设你是一个大型团队的后端程序员,你负责的项目需要使用同事发布在仓库中的依赖,而这些依赖储存在库里但是最终以 Bean 的形式注入到你的代码中的。这个时候为了测试你的代码逻辑,@MockBean 就无法满足你的需求了。而 @SpyBean 便成为了最优雅的解决方案。

使用 @MockBean 和 @SpyBean 解决 SpringBoot 单元测试中 Mock 类装配的问题相关推荐

  1. 解决Springboot+JPA中多表关联查询会查询多次的问题(n+1查询问题)

    解决Springboot+JPA中多表关联查询会查询多次的问题(n+1查询问题) 参考文章: (1)解决Springboot+JPA中多表关联查询会查询多次的问题(n+1查询问题) (2)https: ...

  2. Python——单元测试中mock原理和使用

    摘要 mock主要是的为了提供开发程序员的做一个的单元测试而使用的.假设你开发一个项目,里面包含了一个登录模块,登录模块需要调用身份证验证模块中的认证函数,该认证函数会进行值的返回,然后系统根据这个返 ...

  3. application terminated怎么解决_优雅解决 SpringBoot 工程中多环境下 application.properties 的维护问题...

    个人微信号:geekoftaste, 期待与大家一起探讨! 背景 我们知道 SpringBoot 有一个全局的配置文件 application.properties, 可以把工程里用到的占位符,第三方 ...

  4. 解决SpringBoot+SpringCloud中feign调用服务传递参数为MultipartFile的问题

    文章目录 前言 一.前期说明 二.使用步骤 1.引入maven依赖 2.新建feign的配置类 2.feign客户端 3.被调用的服务的Controller 4.第三方服务远程调用主服务传递Multi ...

  5. 【Maven】一文就解决springboot框架中创建maven所有问题

    声明 以下问题均为我创建项目中所遇到的问题,不一定具有普适性.另外错误案例也没有既是保存,还望海涵 下载Maven 首先进入maven官网,地址在这里 windows选择下载图中这个,然后我们会得到一 ...

  6. java 实体类返回大写_解决springboot bean中大写的字段返回变成小写的问题

    例如我的bean中有以下4个字段 private String code; private String _TOKENUUMS; private String TGC; private String ...

  7. 解决SpringBoot项目中遇到的数据库连接yml文件配置问题

    今天遇到了一个报错 Failed to configure a DataSource: 'url' attribute is not specified and no embedded datasou ...

  8. springboot单元测试中@Autowired自动注入的类一直是null

    另附大佬方法

  9. 单元测试中Assert类的用法

    Assert类所在的命名空间为Microsoft.VisualStudio.TestTools.UnitTesting 在工程文件中只要引用Microsoft.VisualStudio.Quality ...

最新文章

  1. 七个C#编程的小技巧
  2. java对xml文件的读写_java 自己做的对XML文件的读写操作
  3. Django 如何实现 如下 联表 JOIN 查询?
  4. 三十四、R语言数据分析实战
  5. Docke安装MariaDB
  6. 第一篇|腾讯开源项目盘点:WeUI,WePY,Tinker,Mars等
  7. android q测试机型,小米9安卓Q系统刷机包开启测试 小米Android Q适配机型一览
  8. JVM体系结构与工作方式
  9. JAVA 中的 Collection 和 Map 以及相关派生类的概念
  10. 离线CSDN网页打开跳转首页的解决方法
  11. matlab面板数据处理程序,MATLAB空间面板数据模型操作简介
  12. 大数据时代,个人信息安全由谁来保护?
  13. 云计算最终比拼的是什么?
  14. VR眼镜连接android设备,VR眼镜怎么连接手机 VR眼镜使用教程
  15. Intent中putExtra()方法用法
  16. Dynamic Head Unifying Object Detection Heads with Attentions 论文阅读笔记
  17. orb slam [RGBD-1] process has died解决
  18. 第十二期 U-Boot工作原理 《路由器就是开发板》
  19. 点击table中的某一个td,获得这个tr的所有数据
  20. 关于RSA算法的探究 -Crypto 0x01

热门文章

  1. 三种主流深度相机介绍
  2. c++分布式游戏服务器架构设计
  3. Java NIO 系列教程 (十一) Datagram 通道
  4. 2020年10月28日普级组总结
  5. H5ai 更换字体库加快访问速度及相关优化
  6. settings.gradle.kts里读取properties配置文件或者解析json文件
  7. Spring官方文档翻译
  8. SPI,MCP2515调试总结
  9. 服务器电源输出电压不稳定,开关电源输出电压不稳定怎么解决?
  10. 如何快速判断一个数能被整除的方法(1-23之内)