文章目录
前言[^1]
什么是Mockito?
什么是Mock?
为什么要使用Mock?
用PostMan与用Mock有什么区别?
正题
使用Mockito做一个模拟测试
编写代码
开始测试
代码讲解
本节教程结束
前言1
什么是Mockito?
Mockito是GitHub上使用最广泛的Mock框架,并与JUnit结合使用.Mockito框架可以创建和配置mock对象.使用Mockito简化了具有外部依赖的类的测试开发!
什么是Mock?
在面向对象程序设计中,模拟对象(英语:mock object,也译作模仿对象)是以可控的方式模拟真实对象行为的假的对象。程序员通常创造模拟对象来测试其他对象的行为,很类似汽车设计者使用碰撞测试假人来模拟车辆碰撞中人的动态行为。
也就是用来做程序测试的。
为什么要使用Mock?
因为有些时候,你很难再现真实的情况,例如:
真实对象的行为是不确定的(例如,当前的时间或当前的温度);
真实对象很难搭建起来;
真实对象的行为很难触发(例如,网络错误);
真实对象速度很慢(例如,一个完整的数据库,在测试之前可能需要初始化);
真实的对象是用户界面,或包括用户界面在内;
真实的对象使用了回调机制;
真实对象可能还不存在;
真实对象可能包含不能用作测试(而不是为实际工作)的信息和方法。
举一个简单的例子:如果你的程序需要传一个特定时间进行测试,你真的要等到那个时间吗?你不会这样做,而是模拟那个时间传过去
用PostMan与用Mock有什么区别?
经常做测试的肯定对PostMan都不陌生,PostMan是一个专门测试 API 的工具。
为什么我们要选用Mock做测试呢?
首先,PostMan相当于一个客户端(pm),在测试之前你需要先启动你的服务端(srever),相当于是一个客户端的测试
Mock是运行在服务端上的,Mock的作用就是模拟客户端的行为
区别在于Mock省去了网络请求的环节,提高了测试的效率
其次Mock是用编码实现的,PostMan是用真实的网络请求+客户端实现的
还有一个问题,我们在公司中一般都是自动化的单元测试,跑通之后自动打包发布,Mock可以实现但PostMan就不行了
正题
使用Mockito做一个模拟测试
编写代码
1.在Test目录的包下新建一个test包
2.创建一个MockitoTest类,代码如下
package com.moli.zy.springboot2ml.test;
import com.moli.zy.springboot2ml.controller.ArticleRestController;
import lombok.extern.slf4j.Slf4j;
import org.junit.Before;
import org.junit.Test;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.http.HttpMethod;
import org.springframework.test.web.servlet.MockMvc;
import org.springframework.test.web.servlet.MvcResult;
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders;
import org.springframework.test.web.servlet.result.MockMvcResultMatchers;
import org.springframework.test.web.servlet.setup.MockMvcBuilders;
import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print;
@Slf4j
@SpringBootTest
public class MockitoTest {
private MockMvc mockMvc;
@Before
public void setUp() {
mockMvc = MockMvcBuilders.standaloneSetup(new ArticleRestController()).build();
}
@Test
public void saveArticle() throws Exception {
String article = "{\n" +
" \"id\": 1,\n" +
" \"author\": \"moli\",\n" +
" \"title\": \"这是标题\",\n" +
" \"content\": \"这是内容\",\n" +
" \"createTime\": \"2019-08-21 12:30:56\",\n" +
" \"reader\":[{\"name\":\"moli1\",\"age\":18},{\"name\":\"moli2\",\"age\":19}]\n" +
"}";
MvcResult result = mockMvc.perform(MockMvcRequestBuilders.request(HttpMethod.POST, "/rest/article")
.contentType("application/json").content(article))
.andExpect(MockMvcResultMatchers.status().isOk())
.andExpect(MockMvcResultMatchers.jsonPath("$.data.author").value("moli"))
.andExpect(MockMvcResultMatchers.jsonPath("$.data.reader[0].age").value(18))
.andDo(print())
.andReturn();
log.info(result.getResponse().getContentAsString());
}
}
开始测试
可以看到测试没有问题,接下来讲解代码
代码讲解
@SpringBootTest注解
是用来创建Spring的上下文ApplicationContext,保证测试在上下文环境里运行。单独使用@SpringBootTest不会启动servlet容器。不可以使用@Resource和@Autowired等进行依赖注入(准确的说是可以使用,但被注解的bean为null)
在测试方法执行之前,需要先构建一个MockMvc对象,作用就是帮助我们模拟网络请求进行测试
这个方法就是测试的内容了
article变量存的是请求接口时要传输的数据
可以看到 MockMvcRequestBuilders.request 有两个参数,分别是请求的类型,和请求的地址
上图请求的地址对应的Controller方法如图所示
contentType 用于定义发送的数据类型以及发送的对象
这三行是定义我们所期待的结果,如果不符合那么本次测试就会失败
上图三行的详细解释:
第一行:要求状态码必须是200(成功状态), 否则测试失败
第二行:要求返回的数据中data下的author 必须为 moli , 否则测试失败
第三行:要求返回的数据中data下的第一个reader下的age 必须为 18 ,否则测试失败
andDo 表示打印
andReturn 表示将结果返回
本节教程结束
这是我通过学习对知识的整理及备忘,本博客的所有内容,仅是自己的一些学习笔记,如有错误,欢迎指正。如有侵权,请告知修改。
前言部分内容摘选自:https://www.jianshu.com/p/eae0187900f8 ↩︎