在之前的 Spring学习之旅(八)--SpringMVC请求参数 我们是通过在控制台输出来验证参数是否正确,但是这样做实在是太耗时间了,我们今天来学习下 MockMvc,它可以让我们不需要启动项目就能调用接口并验证接口返回结果是否符合我们的预期。

为何使用MockMvc?

MockMvc 实现了对 Http 请求的模拟,能够直接使用网络的形式,实现 Controller 的调用,这样可以使得测试速度快、不依赖网络环境,而且提供了一套验证的工具,这样可以使得请求的验证统一而且很方便。

如何使用 MockMvc

MockMvcBuilder 是用来构造 MockMvc 的构造器,主要有两个实现:

  • StandaloneMockMvcBuilder
  • DefaultMockMvcBuilder

分别对应两种测试方式,即独立安装和集成 Web 环境测试(此种方式并不会集成真正的 web 环境,而是通过相应的 Mock API 进行模拟测试,无须启动服务器)。对于我们来说直接使用静态工厂 MockMvcBuilders 创建即可。

集成Web环境方式

MockMvcBuilders.webAppContextSetup(WebApplicationContext context) :指定 WebApplicationContext,将会从该上下文获取相应的控制器并得到相应的 MockMvc

  1. @RunWith(SpringJUnit4ClassRunner.class)
  2. @ContextConfiguration(classes = RootConfig.class)
  3. @WebAppConfiguration
  4. public class RequestParameterControllerTest {
  5. @Autowired
  6. private WebApplicationContext wac;
  7. private MockMvc mockMvc;
  8. @Before
  9. public void setUp() {
  10. mockMvc = MockMvcBuilders.webAppContextSetup(wac).build();
  11. }
  12. }

独立测试方式

MockMvcBuilders.standaloneSetup(Object... controllers) :通过参数指定一组控制器,这样就不需要从上下文获取了;

  1. @RunWith(SpringJUnit4ClassRunner.class)
  2. @ContextConfiguration(classes = RootConfig.class)
  3. @WebAppConfiguration
  4. public class RequestParameterControllerTest {
  5. @Autowired
  6. private WebApplicationContext wac;
  7. private MockMvc mockMvc;
  8. @Before
  9. public void setUp() {
  10. mockMvc = MockMvcBuilders.standaloneSetup(new RequestParameterController()).build();
  11. }
  12. }

实例

Controller:

  1. @Controller
  2. public class RequestParameterController {
  3. @RequestMapping("/toInt")
  4. @ResponseBody
  5. public int toInt(int value) {
  6. return value;
  7. }
  8. }

单元测试:

  1. @RunWith(SpringJUnit4ClassRunner.class)
  2. @ContextConfiguration(classes = RootConfig.class)
  3. @WebAppConfiguration
  4. public class RequestParameterControllerTest {
  5. @Autowired
  6. private WebApplicationContext wac;
  7. private MockMvc mockMvc;
  8. @Before
  9. public void setUp() {
  10. mockMvc = MockMvcBuilders.webAppContextSetup(wac).build();
  11. }
  12. @Test
  13. public void toInt() throws Exception {
  14. mockMvc.perform(
  15. // 发送 GET 请求
  16. MockMvcRequestBuilders.get("/toInteger?value=3"))
  17. // 判断HTTP响应码
  18. .andExpect(MockMvcResultMatchers.status().isOk())
  19. // 判断返回内容是否是预期值
  20. .andExpect(MockMvcResultMatchers.content().string("3"))
  21. // 输出整个响应结果信息
  22. .andDo(MockMvcResultHandlers.print());
  23. }
  24. }

控制台输出:

  1. MockHttpServletRequest:
  2. HTTP Method = GET
  3. Request URI = /toInteger
  4. Parameters = {value=[3]}
  5. Headers = {}
  6. Handler:
  7. Type = com.marklogzhu.web.controller.RequestParameterController
  8. Async:
  9. Async started = false
  10. Async result = null
  11. Resolved Exception:
  12. Type = null
  13. ModelAndView:
  14. View name = null
  15. View = null
  16. Model = null
  17. FlashMap:
  18. Attributes = null
  19. MockHttpServletResponse:
  20. Status = 200
  21. Error message = null
  22. Headers = {Content-Type=[text/plain;charset=ISO-8859-1], Content-Length=[1]}
  23. Content type = text/plain;charset=ISO-8859-1
  24. Body = 3
  25. Forwarded URL = null
  26. Redirected URL = null
  27. Cookies = []
  28. Process finished with exit code 0

常用对象

MockMvcRequestBuilders

MockMvcRequestBuilders: 用来构建请求。

方法 作用
MockHttpServletRequestBuilder get(String urlTemplate, Object... urlVariables) 发送 GET 请求
MockHttpServletRequestBuilder post(String urlTemplate, Object... urlVariables) 发送 POST 请求
MockHttpServletRequestBuilder put(String urlTemplate, Object... urlVariables) 发送 PUT 请求
MockHttpServletRequestBuilder delete(String urlTemplate, Object... urlVariables) 发送 DELETE 请求
MockHttpServletRequestBuilder options(String urlTemplate, Object... urlVariables) 发送 OPTIONS 请求
MockHttpServletRequestBuilder request(HttpMethod httpMethod, String urlTemplate, Object... urlVariables) 提供自己的Http请求方法及uri模板和uri变量
MockMultipartHttpServletRequestBuilder fileUpload(String urlTemplate, Object... urlVariables) 发送文件上传请求
RequestBuilder asyncDispatch(final MvcResult mvcResult) 创建一个启动异步处理的请求的 MvcResult 进行异步分派的RequestBuilder

ResultActions

调用 MockMvc.perform(RequestBuilder requestBuilder) 后将得到 ResultActions,通过 ResultActions 可以完成如下三件事:

  • ResultActions andExpect(ResultMatcher matcher) :添加验证断言来判断执行请求后的结果是否是预期的;
  • ResultActions andDo(ResultHandler handler) :添加结果处理器,用于对验证成功后执行的动作,如输出下请求/结果信息用于调试;
  • MvcResult andReturn() :返回验证成功后的MvcResult;用于自定义验证/下一步的异步处理;

ResultMatcher

方法 作用
HandlerResultMatchers handler() 请求的 Handler 验证器,比如验证处理器类型/方法名;此处的 Handler 其实就是处理请求的控制器
RequestResultMatchers request() 得到 RequestResultMatchers 验证器
ModelResultMatchers model() 得到模型验证器
ViewResultMatchers view() 得到视图验证器
FlashAttributeResultMatchers flash() 得到 Flash 属性验证
StatusResultMatchers status() 得到响应状态验证器
HeaderResultMatchers header() 得到响应 Header 验证器
CookieResultMatchers cookie() 得到响应 Cookie 验证器
ContentResultMatchers content() 得到响应内容验证器
JsonPathResultMatchers jsonPath(String expression, Object ... args) 得到Json表达式验证器
ResultMatcher jsonPath(String expression, Matcher matcher) 得到Json表达式验证器
XpathResultMatchers xpath(String expression, Object... args) 得到Xpath表达式验证器
XpathResultMatchers xpath(String expression, Map<string, string=""> namespaces, Object... args) 得到Xpath表达式验证器
ResultMatcher forwardedUrl(final String expectedUrl) 验证处理完请求后转发的url(绝对匹配)
ResultMatcher forwardedUrlPattern(final String urlPattern) 验证处理完请求后转发的url(Ant风格模式匹配,@since spring4)
ResultMatcher redirectedUrl(final String expectedUrl) 验证处理完请求后重定向的url(绝对匹配)
ResultMatcher redirectedUrlPattern(final String expectedUrl) 验证处理完请求后重定向的url(Ant风格模式匹配,@since spring4)

实例

JSON请求/响应验证

  1. String requestBody = "{\"id\":1, \"name\":\"zhang\"}";
  2. mockMvc.perform(post("/user")
  3. .contentType(MediaType.APPLICATION_JSON).content(requestBody)
  4. .accept(MediaType.APPLICATION_JSON)) //执行请求
  5. .andExpect(content().contentType(MediaType.APPLICATION_JSON)) //验证响应contentType
  6. .andExpect(jsonPath("$.id").value(1)); //使用Json path 验证JSON,具体表达式规则请参考 http://goessner.net/articles/JsonPath/
  7. String errorBody = "{id:1, name:zhang}";
  8. MvcResult result = mockMvc.perform(post("/user")
  9. .contentType(MediaType.APPLICATION_JSON).content(errorBody)
  10. .accept(MediaType.APPLICATION_JSON)) //执行请求
  11. .andExpect(status().isBadRequest()) //400错误请求
  12. .andReturn();
  13. Assert.assertTrue(HttpMessageNotReadableException.class.isAssignableFrom(result.getResolvedException().getClass()));//错误的请求内容体

文件上传

  1. byte[] bytes = new byte[] {1, 2};
  2. mockMvc.perform(fileUpload("/user/{id}/icon", 1L).file("icon", bytes)) //执行文件上传
  3. .andExpect(model().attribute("icon", bytes)) //验证属性相等性
  4. .andExpect(view().name("success")); //验证视图

自定义验证

  1. MvcResult result = mockMvc.perform(get("/user/{id}", 1))//执行请求
  2. .andReturn(); //返回MvcResult
  3. Assert.assertNotNull(result.getModelAndView().getModel().get("user")); //自定义断言

Spring学习之旅(十)--MockMvc的更多相关文章

  1. Spring学习之旅(十五)--SpringBoot

    在使用 Spring 的过程中,有时候会出现一些 ClassNotFoundException 异常,这是因为 JAR 依赖之间的版本不匹配所导致的.而 Spring Boot 就能避免绝大多数依赖版 ...

  2. Spring学习之旅(十四)--缓存

    数据库的读写并发一直都是应用性能的瓶颈所在之一,针对改动频率很小的数据我们应该将他存放到缓存中,减少与数据库的交互. 启用对缓存的支持 Spring 对缓存的支持有两种方式: 注解驱动的缓存 XML ...

  3. Spring学习之旅(十二)--持久化框架

    对于本职工作来说 JDBC 就可以很好的完成,但是当我们对持久化的需求变得更复杂时,如: 延迟加载 预先抓取 级联 JDBC 就不能满足了,我们需要使用 ORM框架 来实现这些需求. Spring 对 ...

  4. Java框架spring 学习笔记(十八):事务管理(xml配置文件管理)

    在Java框架spring 学习笔记(十八):事务操作中,有一个问题: package cn.service; import cn.dao.OrderDao; public class OrderSe ...

  5. Spring学习之旅(八)Spring 基于AspectJ注解配置的AOP编程工作原理初探

    由小编的上篇博文可以一窥基于AspectJ注解配置的AOP编程实现. 本文一下未贴出的相关代码示例请关注小编的上篇博文<Spring学习之旅(七)基于XML配置与基于AspectJ注解配置的AO ...

  6. Spring学习之旅(六)--SpringMVC集成

    对大多数 Java 开发来说,基于 web 的应用程序是我们主要的关注点. Spring 也提供了对于 web 的支持,基于 MVC 模式的 Spring MVC 能够帮助我们灵活和松耦合的完成 we ...

  7. Java框架spring 学习笔记(十四):注解aop操作

    回见Java框架spring Boot学习笔记(十三):aop实例操作,这里介绍注解aop操作 首先编写一个切入点HelloWorld.java package com.example.spring; ...

  8. Spring学习之旅(十三)--使用NoSQL数据库

    除了关系型数据库之外,现在还有一种 NoSQL 数据库非常流行,而 Spring 自然也没有放过对它的支持. NoSQL 数据库有很多种,如: MongoDBGenericJackson2JsonRe ...

  9. Spring学习之旅(三)--装配Bean

    装配 Bean 的方式 在 XML 中进行显式配置 在 Java 中进行显式配置 隐式的 Bean 发现机制和自动装配 Spring 提供了以上三种方式进行 Bean 的配置,可以根据自己的需求选择一 ...

随机推荐

  1. Python小故事--------Tkinter的组件描述及解析

    概念 Tkinter: 是Tk图形用户界面工具包标准(ctl)的Python接口,作为一个轻量级的跨平台图形用户界面(GUI)开发工具 frame: 屏幕上的一块矩形区域,多是用来作为容器(conta ...

  2. python面向对象的继承-组合-02

    *面向对象(OOP)的三大特征:**# 封装.继承.多态 继承 什么是继承 继承:# 是一种关系,描述两个对象之间什么是什么的什么的关系 例如:麦兜.佩奇.猪猪侠.猪刚鬣,都是猪 为什么要使用继承 继 ...

  3. golang "[]uint8" to string

    关于Uinit8和Byte: The Go Programming Language Specification Numeric types uint8 the set of all unsigned ...

  4. iis8 php-cgi.exe - FastCGI 进程意外退出 500错误解决办法

    今天iis服务环境下的网站突然显示200错误php-cgi.exe - FastCGI 进程意外退出,昨天还好好的网站正常,这个问题一直偶尔出现几次,不是很频繁,但是偶尔会出现: 这是由于某些加载库加 ...

  5. JAVA遍历机制的性能的比较

        本文首发于cartoon的博客     转载请注明出处:https://cartoonyu.github.io/cartoon-blog/post/java/java%E9%81%8D%E5% ...

  6. bootstrap datatable editor 扩展

    需求: a. 表单样式更改. b. 表单大小更改. 思路: a. 通过设置modal css更改样式和大小.缺点,全局性的更改. b. 更改bootstrap-editor,可以通过某种方式将参数传入 ...

  7. spring aop(四)

    直接找到解析aop标签的方法: protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate deleg ...

  8. (技能篇)双机热备之Oracle切换故障处理

    背景: 以前做的的一个项目中使用了某国产双机热备产品,但是在数据库做双机热备时出现了一些问题,没办法.不得不研究一番了!经过两天的研究终于问题得以解决.将问题处理步骤记录下来以备后用,也希望能帮助到需 ...

  9. el-upload自定义上传文件,并携带其余参数,且action不报错

    用el-upload组件自定义上传按钮,并携带其余参数,且必传参数action 不报错 <template> <el-col :span="6" :mode=&q ...

  10. 【iOS】No suitable application records found

    昨天提交 Apple 审核时遇到这个问题,如图: 原来是还没在 iTunes Connect 创建 APP ... 一时着急大意了…… 后来想想还真是脑子一时没反应过来……