[本文出自天外归云的博客园]

概要简述

利用JUnit结合Mockito,再加上spingframework自带的一些方法,就可以组合起来对Spring MVC中的Controller层进行测试。

在设计测试用例前,我们要对待测Controller的代码逻辑进行逐层深入的走查。走查的目的是要明确Controller中主要逻辑分支,以便设计测试用例进行覆盖。一些主要通用的关注点有:

1. 请求request中所包含的参数值(Controller中从请求中获取的参数)

2. Controller中的try块中能够引起异常的方法调用

3. Controller中的if语句涉及的变量值

4. 一些ThreadLocal方法(在实际测试过程中需要对ThreadLocal对象做一些操作来模拟一些状态)

测试规范

创建后端测试分支:一定是以开发分支为基础创建,也为通过修改开发代码来调试测试代码创造方便。

创建测试类:测试类名B与待测类名A的关系为B=ATest

测试类上添加注释:@RunWith(MockitoJUnitRunner.class)

测试类中声明:private MockMvc mockMvc;

测试类中待注入mock的对象声明上添加注释:@InjectMocks

测试类中待mock的对象声明上添加注释:@Mock

测试方法上添加注释:@Test

常用引用(如果IDE不能自动下载对应maven仓库,则需手动修改pom.xml文件添加引用相应的maven仓库):

import org.junit.Before;
import org.junit.Test;
import org.junit.runner.JUnitCore;
import org.junit.runner.Result;
import org.junit.runner.RunWith;
import org.junit.runner.notification.Failure;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import org.mockito.runners.MockitoJUnitRunner;
import org.springframework.test.web.servlet.MockMvc;
import org.springframework.test.web.servlet.RequestBuilder;
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders;
import org.springframework.test.web.servlet.setup.MockMvcBuilders; import static org.mockito.Matchers.any;
import static org.mockito.Matchers.anyString;
import static org.mockito.Mockito.doReturn;
import static org.mockito.Mockito.doThrow;
import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;

初始化方法标准范例:

@Before
public void setUp() throws Exception {
MockitoAnnotations.initMocks(this);
this.mockMvc = MockMvcBuilders.standaloneSetup(声明的controller).build();
}

测试方法标准结构范例:

@Test
public void testSth() throws Exception {
// 自定义填充
// Mock依赖
// 构造请求
// 执行请求与断言
}

对环境ThreadLocal的填充方法举例:

//填充为空
XXThreadLocalContainer.XXX_THREAD_LOCAL.set(null);
//填充为指定类型对象
A a= new A();
a.setXX("test");
XXThreadLocalContainer.XXX_THREAD_LOCAL.set(a);

对无返回service方法依赖注入的mock方法举例:

doThrow(Exception.class).when(someService).doSomeMethod(any(SomeClassA.class), any(SomeClassB.class), anyString(), anyString(), any(SomeEnum.class));

对有返回service方法依赖注入的mock方法举例:

// 构造mock方法返回的对象
A a = new A();
a.setSomePropertyA(someValueA);
a.setSomePropertyB(someValueB);
// 构造mock方法
doReturn(a).when(someService).doSomeMethod(anyString(), anyString());

利用RequestBuilder构造GET请求举例:

RequestBuilder request = MockMvcRequestBuilders.get(someUrl).requestAttr("someAttrNameA", someValueA).requestAttr("someAttrNameB", someValueB).requestAttr("someAttrNameC", someValueC);

执行请求与断言举例:

mockMvc.perform(request).andDo(print()).andExpect(jsonPath("someFieldName").value(String.valueOf(someFieldValue)));

自定义main函数执行测试:

public static void main(String[] args) {
Result result = JUnitCore.runClasses(MpResurrectionControllerTest.class);
for (Failure failure : result.getFailures()) {
System.out.println(String.format("FAILED : %s", failure.toString()));
}
System.out.println(String.format("TEST SUCCESS : %s", result.wasSuccessful()));
}

抽象复用

在实际的测试过程中,把复用的部分提取抽象,生成一个基类为测试类提供继承(MockTestBase.java),其中testAll方法利用反射通过类名动态生成类对象:

package com.xx.xxx;

import org.junit.Before;
import org.junit.Rule;
import org.junit.rules.ExpectedException;
import org.junit.runner.JUnitCore;
import org.junit.runner.Result;
import org.junit.runner.RunWith;
import org.junit.runner.notification.Failure;
import org.mockito.MockitoAnnotations;
import org.mockito.runners.MockitoJUnitRunner;
import org.springframework.test.web.servlet.MockMvc; /**
* @Author: Tylan
* @CreateDate: 2018/7/2 11:07
* @UpdateDate: 2018/7/2 11:07
*/
@RunWith(MockitoJUnitRunner.class)
public abstract class MockTestBase { MockMvc mockMvc; @Rule
public ExpectedException thrown = ExpectedException.none(); @Before
public void setUp() {
MockitoAnnotations.initMocks(this);
} public static void testAll(String className) throws ClassNotFoundException {
Class obj = Class.forName(className);
Result result = JUnitCore.runClasses(obj);
for (Failure failure : result.getFailures()) {
System.out.println(String.format("FAILED : %s", failure.toString()));
}
System.out.println(String.format("TEST SUCCESS : %s", result.wasSuccessful()));
}
}

新的测试类就变成这样,重新写一个main函数调用基类testAll方法,利用反射传入当前运行的类名:

package com.xx.xxx;

import com.xx.xxx.ForTestController;
import com.xx.xxx.SomeService;
import org.junit.Test;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.springframework.test.web.servlet.RequestBuilder;
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders;
import org.springframework.test.web.servlet.setup.MockMvcBuilders; import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath; /**
* @Author: Tylan
* @CreateDate: 2018/7/2 11:11
* @UpdateDate: 2018/7/2 11:11
*/
public class NewMpGameOrderControllerTestsTylan extends MockTestBase {
@InjectMocks
private ForTestController someController; @Mock
SomeService someService; @Override
public void setUp() {
this.mockMvc = MockMvcBuilders.standaloneSetup(someController).build();
} @Test
public void insertTestWrongParam() throws Exception {
/*
* 测试data为空
* */
System.out.println("insertTestWrongParam");
// 构造测试数据
String data1 = "";
String data2 = "";
String insertUrl = "/xx/xx/data/insert";
// 执行请求与断言
RequestBuilder request = MockMvcRequestBuilders.get(insertUrl).requestAttr("data1", data1).requestAttr("data2", data2);
mockMvc.perform(request).andDo(print()).andExpect(jsonPath("xxx").value("xx"));
} public static void main(String[] args) throws ClassNotFoundException {
testAll(Thread.currentThread().getStackTrace()[1].getClassName());
}
}

这就生成了一个测试Controller的模板。剩下的工作就是分析开发源码,设计测试用例并对Controller中的所有逻辑分支进行覆盖测试了。

在一个单元测试用例,标准顺序是mock、test、verify三个环节。其中mock举例:

GameOrder order = new GameOrder();
order.setId(xxx);
order.setScore(xxx); doReturn(order).when(gameService).initGameOrder(anyString(), anyString(), anyString(), any(MpEnum.class));

涉及对foreach循环的mock举例:

Iterator<SomeService> mockIter = mock(Iterator.class);
when(mockIter.hasNext()).thenReturn(true, true, true, false);
when(mockIter.next()).thenReturn(someService).thenReturn(someService).thenReturn(someService);
when(serviceList.iterator()).thenReturn(mockIter);
doReturn(SomeEnum.XXXXX).when(someService).getSomeEnum();
doReturn(true).when(someService).isValid(anyMap());

对于test环节,测试Controller就是模拟给Controller发请求,测试Service就是直接调用Service方法。

对于Verify环节,主要是对一些方法是否执行,路径是否走过做一下验证,以及一些值的验证。这里举个例子:

//Verify
verify(someService).initGameOrder(xx, xxx, xxxx, someEnum);
verify(someService).update(any(GameOrder.class));

这里验证了someService对象是否执行了initGameOrder方法和update方法。

感受与总结

想做好单元测试,要对待测试代码逻辑进行充分分析,重点逻辑是在Service层和Controller层的测试,而对于这两层来说,Controller中除了一些基本逻辑基本就是service层的调用,对于service层,除了定义service的interface类就是定义对应implements的接口实现类,interface类只定义方法接口,剩余的由service的impl类实现,service实现类中除了Override实现继承的接口类定义的方法接口就是Autowired声明一些在该实现类中要用到的其他service接口类(非实现类),剩下的就是接口类一层又一层的继承关系,而service层无非是围绕着dao层展开的,最终落在了dao层对数据库或缓存的操作上。所以纵观整个后端结构顺序就是Filter层-Controller层-Service层-Dao层,请求是按这个顺序前进与原路返回的。所以后端测试的重点,最终是落在Controller层逻辑的正确性与sql查询、redis和memcache等缓存查询的正确性上。

单测是否应该由开发人员进行?这个问题的答案和开发是否应该由测试人员完成是一样的,如果让开发搞测试没什么不可以,那么让测试做开发也能更好的保障质量。但是现实的分工中,没有那么多全栈人才,测试人员不懂开发,开发人员不懂测试。单测是应该由测试人员完成的,想要保障产品质量,如果QA都对代码逻辑没有了解的话,单凭瞎子摸象这种经验推测式的方法进行测试,能够发现的问题种类和层次也是有限的。而开发除了紧张的开发周期外,还要修改bug,没有更多的时间来投入到测试中。所以测试人员应该具备能够发现bug和定位问题原因的能力才能够更好的配合开发一起完成高质量的产品软件工程。

JUnit+Mockito结合测试Spring MVC Controller的更多相关文章

  1. 使用MockMvc测试Spring mvc Controller

    概述   对模块进行集成测试时,希望能够通过输入URL对Controller进行测试,如果通过启动服务器,建立http client进行测试,这样会使得测试变得很麻烦,比如,启动速度慢,测试验证不方便 ...

  2. 就是这么简单(续)!使用 RestAssuredMockMvc 测试 Spring MVC Controllers

    就是这么简单(续)!使用 RestAssuredMockMvc 测试 Spring MVC Controllers 转载注明出处:http://www.cnblogs.com/wade-xu/p/43 ...

  3. Spring MVC Controller 单元测试

    简介 Controller层的单元测试可以使得应用的可靠性得到提升,虽然这使得开发的时间有所增加,有得必失,这里我认为得到的比失去的多很多. Sping MVC3.2版本之后的单元测试方法有所变化,随 ...

  4. Spring MVC Controller与jquery ajax请求处理json

    在用 spring mvc 写应用的时候发现jquery传递的[json数组对象]参数后台接收不到,多订单的处理,ajax请求: "}]}]} $.ajax({ url : url, typ ...

  5. 关于Spring MVC Controller 层的单元测试

    关于Spring MVC Controller 层的单元测试 测试准备工作: 1.搭建测试Web环境 2.注入Controller 类 3.编写测试数据 测试数据的文件名一定要与测试类的文件名相同,比 ...

  6. Spring MVC Controller中解析GET方式的中文参数会乱码的问题(tomcat如何解码)

    Spring MVC Controller中解析GET方式的中文参数会乱码的问题 问题描述 在工作上使用突然出现从get获取中文参数乱码(新装机器,tomcat重新下载和配置),查了半天终于找到解决办 ...

  7. 转:【Spring MVC Controller单例陷阱】

    http://lavasoft.blog.51cto.com/62575/1394669/ Spring MVC Controller默认是单例的: 单例的原因有二:1.为了性能.2.不需要多例. 1 ...

  8. Spring MVC Controller单例陷阱

    原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://lavasoft.blog.51cto.com/62575/1394669 Spr ...

  9. Spring MVC Controller中GET方式传过来的中文参数会乱码的问题

    Spring MVC controller 这样写法通常意味着访问该请求,GET和POST请求都行,可是经常会遇到,如果碰到参数是中文的,post请求可以,get请求过来就是乱码.如果强行对参数进行了 ...

随机推荐

  1. HDU 2795 Billboard【点修改】

    <题目链接> 题目大意: 有一块h*w的矩形广告板,要往上面贴广告,然后给n个1*wi的广告,要求把广告贴上去,而且要求广告要尽量往上贴并且尽量靠左, 求第n个广告的所在的位置,不能贴则为 ...

  2. 关于Maven打包Java Web项目以及热部署插件Jrebel的使用

    Java Web/Eclipse/Maven/Tomcat 最近有个新项目是java web项目,记录一下,可能比较乱.虽然没接触过Java,但是eclipse还是用过的 初识项目 同事说,项目是ma ...

  3. 上线---苹果AppStore审核注意事项,Guideline 1.2 - Safety - User Generated Content,2.1等条例(苹果审核六次拒绝)

    前段时间上线app,和战友一起撸了那么久的代码,上线是最激动的.然而安卓各大平台上线了半个月了,苹果却给了六次拒绝. 刚开始等苹果等的焦头烂额,现在内心毫无波澜,目前还在审核中...... 六次的拒绝 ...

  4. (原创)PouchDB 图片本地存储(web离线应用)

    /* * 参数 * db: 已建或未建数据库 * pouchId: 数据库唯一的主键_id * src: 图片img的DOM对象 * bg: 判断是否是背景图 * */ function addTod ...

  5. java 同步 synchronized

    http://www.cnblogs.com/Qian123/p/5691705.html http://www.cnblogs.com/GnagWang/archive/2011/02/27/196 ...

  6. Eclipse 重构

    使用重构工具可以快捷地将代码变成整洁而高度模块化的代码. Rename Rename 是 Eclipse 中最常用的重构.利用这个重构,可以对变量.类.方法.包.文件夹及几乎任何的 Java 标识符进 ...

  7. 几种Unity运行平台的判断

    这里就介绍几种常见的,也是便于使用的几种平台判断的方法. 1.先说第一种,也是我用的顺手的一个.利用RuntimePlatform判断,API上的解释是[The platform applicatio ...

  8. Java中的ReentrantLock和synchronized两种锁定

    原文:http://www.ibm.com/developerworks/cn/java/j-jtp10264/index.html 多线程和并发性并不是什么新内容,但是 Java 语言设计中的创新之 ...

  9. 版本视图找不到数据 EDITIONING VIEW

    Oracle database 12 以后的版本,特别在EBS R12.2.X加入了版本视图这种技术,跟MOAC有点像. CREATE OR REPLACE FORCE EDITIONING VIEW ...

  10. WinPcap权威指南(一)

    WinPcap是一个开源的网络抓包模块,顾名思义,它只能工作在Windows下,但本文介绍的知识并不局限于任何操作系统和开发语言,因为网络协议本身是没有这些区别的.阅读本指南之前,请先下载WinPca ...