给controller写单元测试的方法有主要有两种
方法一:
依赖service以及dao层,通过内存数据库H2或者dbUnit做假数据库,insert假数据到内存中
@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration
@ContextConfiguration(locations = "classpath:applicationContext-test.xml")
public abstract class ControllerTestCase extends TestCase {
protected MockMvc mockMvc;
@Before
public void setUp() throws Exception {
mockMvc = MockMvcBuilders.standaloneSetup(getController()).build();
}
@Test
public void testSave() throws Exception {
ResultActions resultActions = this.mockMvc.perform(MockMvcRequestBuilders.put("/test/save.do")
.accept(MediaType.APPLICATION_JSON).param("id", "1000").param("name", "test"));
MvcResult mvcResult = resultActions.andReturn();
ModelAndView modelAndView = mvcResult.getModelAndView();
System.out.println(modelAndView.getViewName());
assertEquals("redirect:null", modelAndView.getViewName());
}
}
方法二:
这种方法是基于mock,用mockito创建mock对象,解决controller对其他类的依赖问题,我倾向于用这种方法,因为这种方法,脱离了类之间的依赖关系,实现了真正意义上的组件的单元测试。同时我也避免了写数据库sql到内存等一些了啰嗦的问题。
demo如下:
public class CreativeTemplateControllerTest extends TestCase{
private CreativeTemplateController creativeTemplateController;
private CreativeTemplateService mockCreativeTemplateService;
@Before
public void setUp() throws Exception {
creativeTemplateController = new CreativeTemplateController();
mockCreativeTemplateService= mock(CreativeTemplateService.class);
creativeTemplateController.setCreativeTemplateService(mockCreativeTemplateService);
}
@Test
public void testSave() {
MockHttpServletRequest request = new MockHttpServletRequest("POST", "/save.do");
request.addParameter("url", "index?pageNo=20");
request.setSession(new MockHttpSession(null));
HttpServletResponse response = new MockHttpServletResponse();
CrtTemplate crtTemplate = new CrtTemplate();
crtTemplate.setId(1L);
crtTemplate.setName("test");
try {
String result = creativeTemplateController.save(request, response, crtTemplate);
assertEquals("redirect:index?pageNo=20", result);
verify(mockCreativeTemplateService).saveCrtTemplate(crtTemplate);
} catch (UnsupportedEncodingException e) {
fail();
}
}
这种方法还可能会用到spring的一些mock类:
MockHttpServletRequest—几乎每个单元测试中都要使用这个类,它是J2EE Web应用程序最常用的接口HttpServletRequest的mock实现。
MockHttpServletResponse—此对象用于HttpServletResponse接口的mock实现。
MockHttpSession—这是另外一个经常使用的mock对象(后文将讨论此类在会话绑定处理中的应用)。
DelegatingServletInputStream—这个对象用于ServletInputStream接口的mock实现。
DelegatingServletOutputStream—这个对象将代理ServletOutputStream实现。在需要拦截和分析写向一个输出流的内容时,你可以使用它。
总之,在实现你自己的测试控制器时,上面这些对象是最为有用的。然而,Spring也提供了下列相应于其它不太常用的组件的mock实现(如果你是一个底层API开发者,那么你可能会找到其各自的相应用法):
MockExpressionEvaluator—这个mock对象主要应用于你想开发并测试你自己的基于JSTL的标签库时。
MockFilterConfig—这是FilterConfig接口的一个mock实现。
MockPageContext—这是JSP PageContext接口的一个mock实现。你会发现这个对象的使用有利于测试预编译的JSP。
MockRequestDispatcher—RequestDispatcher接口的一个mock实现,你主要在其它mock对象内使用它。
MockServletConfig—这是ServletConfig接口的一个mock实现。在单元测试某种Web组件(例如Struts框架所提供的Web组件)时,要求你设置由MockServletContext所实现的 ServletConfig和ServletContext接口。
那么,我们该如何使用这些mock对象呢?我们知道,HttpServletRequest 是一个持有描述HTTP参数的固定值的组件,而正是这些参数驱动Web组件的功能。MockHttpServletRequest,作为 HttpServletRequest接口的一个实现,允许你设置这些不可改变的参数。在典型的Web组件测试情形下,你可以实例化这个对象并按如下方式设置其中的任何参数:
//指定表单方法和表单行为
MockHttpServletRequest request = new MockHttpServletRequest("GET", "/main.app");
request.addParameter("choice", expanded);request.addParameter("contextMenu", "left");
同样地,你可以实例化并全面地控制和分析HttpResponse和HttpSession对象。接下来,让我们简要观察Spring所提供的特定的JUnit框架扩展。