SpringMVC框架如何與Junit整合看這個(gè)就夠了
系統(tǒng)環(huán)境
軟件 | 版本 |
---|---|
spring-webmvc | 4.3.6.RELEASE |
spring-test | 4.3.6.RELEASE |
junit | 4.12 |
引入依賴
<dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> <scope>test</scope> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>${spring.version}</version> <scope>test</scope> </dependency>
如上,引入了4個(gè)依賴,其中:
- junit 是Junit單元測(cè)試的基礎(chǔ)依賴,它提供了@RunWith和@Test等單元測(cè)試必備的注解。<scope>test</scope> 則指明了該依賴的作用范圍是test階段。
- spring-test依賴可以理解成是spring框架與junit框架的橋梁。提供了@WebAppConfiguration等注解。
編寫單元測(cè)試基類
@RunWith(SpringJUnit4ClassRunner.class) @WebAppConfiguration @ContextConfiguration("file:src/test/java/spring-config.xml") @ActiveProfiles("dev") @TransactionConfiguration(transactionManager = "transactionManager", defaultRollback = true) public abstract class BaseControllerTest { protected MockMvc mvc; @Autowired private WebApplicationContext context; //獲取系統(tǒng)上下文 protected WebApplicationContext getCtx() { return this.context; } //初始化方法 @Before public void setup() throws SQLException { this.mvc = MockMvcBuilders.webAppContextSetup(this.context).build(); } //封裝返回結(jié)果 protected String getMockResultStr(MockHttpServletRequestBuilder requestBuilder) { try { MvcResult mvcResult = this.mockMvc .perform(requestBuilder) .andExpect(status().isOk()) .andDo(MockMvcResultHandlers.print()) .andReturn(); MockHttpServletResponse response = mvcResult.getResponse(); response.setContentType(MediaType.APPLICATION_JSON_UTF8_VALUE); return response.getContentAsString(); } catch (Exception e) { e.printStackTrace(); return null; } } }
代碼說明
- @RunWith(SpringJUnit4ClassRunner.class) 注解表明使用SpringJUnit4ClassRunner來運(yùn)行測(cè)試用例。
- @WebAppConfiguration 注解表明使用的ApplicationContext將是WebApplicationContext類型的;value屬性指定web應(yīng)用的根;同時(shí),他必須與 @ContextConfiguration一同使用。
- @ContextConfiguration注解的作用是導(dǎo)入我們需要的配置文件或者類,這里指定了SpringMVC的配置文件
- @ActiveProfiles 注解是指定單元測(cè)試連接的環(huán)境。
- @TransactionConfiguration 注解是用與管理spring事務(wù)的。
- MockHttpServletRequest 類為了模擬HttpServletRequest 對(duì)象,可以通過它來設(shè)置請(qǐng)求頭,這在需要鑒權(quán)的接口測(cè)試中十分實(shí)用。
- MockMvc 實(shí)現(xiàn)了對(duì)Http請(qǐng)求的模擬,能夠直接使用網(wǎng)絡(luò)的形式,轉(zhuǎn)換到Controller的調(diào)用,這樣可以使得測(cè)試速度快,不依賴網(wǎng)絡(luò)環(huán)境,而且提供了一套驗(yàn)證的工具,這樣可以使得請(qǐng)求的驗(yàn)證統(tǒng)一而且很方便。
getMockResultStr 方法是封裝了公共的測(cè)試方法以及響應(yīng)參數(shù)。 其中:
MvcResult mvcResult = this.mockMvc .perform(requestBuilder) .andExpect(status().isOk()) .andDo(MockMvcResultHandlers.print()) .andReturn();
- perform方法是執(zhí)行一個(gè)RequestBuilder請(qǐng)求,調(diào)用Controller的業(yè)務(wù)處理邏輯。
- andExpect 方法是添加執(zhí)行完成后的斷言,添加ResultMatcher驗(yàn)證規(guī)則,驗(yàn)證控制器執(zhí)行完成后結(jié)果是否正確,此處是驗(yàn)證狀態(tài)是否是200
- andDo 方法是添加一個(gè)結(jié)果處理器,比如再此處是andDo(MockMvcResultHandlers.print())輸出整個(gè)響應(yīng)結(jié)果信息,可以在調(diào)試的時(shí)候使用。
- andReturn:方法是執(zhí)行完成后返回相應(yīng)的結(jié)果。
MockMvcRequestBuilders
- RequestBuilder是用來構(gòu)建請(qǐng)求的。其提供了一個(gè)方法buildRequest(ServletContext servletContext) 用于構(gòu)建MockHttpServletRequest; 其主要有兩個(gè)子類MockHttpServletRequestBuilder和MockMultipartHttpServletRequestBuilder(文件上傳時(shí)使用),即用來mock客戶端請(qǐng)求需要的所有數(shù)據(jù)。
- 主要API有MockHttpServletRequestBuilder.get(String urlTemplate, Object… urlVariables): 根據(jù)uri模板和uri變量值得到一個(gè)GET請(qǐng)求方式的RequestBuilder,如果在controller的方法中method選擇的是RequestMethod.GET,那在controllerTest中對(duì)應(yīng)就要使用MockMvcRequestBuilders.get。
- MockHttpServletRequestBuilder.post(String urlTemplate, Object… urlVariables):同get類似,但是是post方法,通過param 方法來設(shè)置請(qǐng)求參數(shù)。
返回結(jié)果的content-type是application/json的請(qǐng)求的單元測(cè)試
get請(qǐng)求測(cè)試
get請(qǐng)求的測(cè)試就是調(diào)用MockMvcRequestBuilders的get方法, 在該方法中傳入接口地址。
MockHttpServletRequestBuilder requestBuilder = MockMvcRequestBuilders.get("/v1/works/lineup/set") .param("id", "123") .param("first_flag", String.valueOf(true)); String result = getMockResultStr(requestBuilder);
post請(qǐng)求測(cè)試
post請(qǐng)求的測(cè)試就是調(diào)用MockMvcRequestBuilders的post方法, 在該方法中傳入接口地址。
MockHttpServletRequestBuilder requestBuilder = MockMvcRequestBuilders.post("/v1/blockly/copy") .param("id", "1211"); success(requestBuilder);
文件上傳測(cè)試
文件上傳的測(cè)試就是調(diào)用MockMvcRequestBuilders的multipart方法, 在該方法中傳入接口地址。
File file = new File(RestArmWorksControllerTest.class.getClassLoader().getResource("20201102172053.png").toURI()); MockMultipartFile mockMultipartFile = new MockMultipartFile("file", "20201102172053.png", "", new FileInputStream(file)); MockMultipartHttpServletRequestBuilder servletRequestBuilder = MockMvcRequestBuilders.multipart("/creation/fileUpload.do") .file(mockMultipartFile); String result = getMockResultStr(servletRequestBuilder);
發(fā)送一個(gè)json格式的數(shù)據(jù)到Controller層
String requestJson = JSONObject.toJSONString(folderInfo); MockHttpServletRequestBuilder requestBuilder = MockMvcRequestBuilders.post("http://v1/works/save").contentType(MediaType.APPLICATION_JSON).content(requestJson);
返回結(jié)果是視圖請(qǐng)求的單元測(cè)試
那么,問題來了,如果一個(gè)接口的返回結(jié)果是視圖,那么這個(gè)接口的單元測(cè)試該如何寫呢?比如下面這個(gè)接口:
@RequestMapping(value = "/edit") public String edit(ModelMap model, HttpServletRequest request, HttpServletResponse response) { Long scriptId = null; if (StringUtils.isNotBlank(request.getParameter("scriptId"))) { scriptId = Long.parseLong(request.getParameter("scriptId")); } if (scriptId != null) { MallSuperscript mallSuperscript = superScriptService.getScriptById(scriptId); model.addAttribute("superScript", mallSuperscript); } model.addAttribute("scriptTypeMap", ScriptType.toMap()); logger.info("角標(biāo)編輯頁(yè)面返回成功"); return "manage/superScript/edit"; }
這個(gè)接口返回的是一個(gè)視圖,就是名稱為edit.jsp頁(yè)面。它的測(cè)試方法可以像下面這樣寫。
@Test public void testEdit() throws Exception { MvcResult mvcResult = mvc .perform( get("/manage/tab2/superScript/edit").sessionAttr("adminid", "1111") .sessionAttr("platformid", 1001L).param("scriptId", "1111")) .andExpect(status().isOk()).andDo(print()).andReturn(); assertEquals("manage/superScript/edit", mvcResult.getModelAndView().getViewName()); }
通過mvcResult.getModelAndView().getViewName()獲取視圖的名稱。
直接測(cè)試一個(gè)service的方法
上面的那些測(cè)試針對(duì)的都是Controller層的接口,那么如果要測(cè)試一個(gè)service層的方法,該如何處理呢?
public class MallMailJobTest extends BaseControllerTest { private MallMailJob mallMailJob; @Override @Before public void setup() throws SQLException { super.setup(); mallMailJob = getCtx().getBean(MallMailJob.class); } /** * * @throws Exception */ @Test public void testExecute() throws Exception { mallMailJob.execute(); }
被測(cè)試的MallMailJob類也可以通過@Autowired 注解直接注入進(jìn)Spring IOC容器中。拿到被測(cè)試類的實(shí)例之后就可以直接調(diào)用被測(cè)試的方法,是不是灰常的簡(jiǎn)單呀。
總結(jié)
本文詳細(xì)介紹在SpringMVC任何使用Junit框架。首先介紹了如何引入依賴,接著介紹了編寫一個(gè)測(cè)試基類,并且對(duì)其中涉及的各個(gè)注解做了一個(gè)詳細(xì)說明。最后就是羅列了各種測(cè)試情況。
到此這篇關(guān)于SpringMVC框架如何與Junit整合看這個(gè)就夠了的文章就介紹到這了,更多相關(guān)SpringMVC整合Junit內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
超簡(jiǎn)單的java獲取鼠標(biāo)點(diǎn)擊位置坐標(biāo)的實(shí)例(鼠標(biāo)在Jframe上的坐標(biāo))
在Java窗體Jframe上獲取鼠標(biāo)點(diǎn)擊的坐標(biāo),其中使用了匿名內(nèi)部類,實(shí)例代碼非常簡(jiǎn)單易懂,大家可以學(xué)習(xí)一下2018-03-03Java帶復(fù)選框的樹(Java CheckBox Tree)實(shí)現(xiàn)和應(yīng)用
這篇文章主要為大家詳細(xì)介紹了Java帶復(fù)選框的樹實(shí)現(xiàn)和應(yīng)用,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11Springboot中使用攔截器、過濾器、監(jiān)聽器的流程分析
Javaweb三大組件:servlet、Filter(過濾器)、?Listener(監(jiān)聽器),這篇文章主要介紹了Springboot中使用攔截器、過濾器、監(jiān)聽器的流程分析,感興趣的朋友跟隨小編一起看看吧2023-12-12基于mybatis-plus QueryWrapper 排序的坑
這篇文章主要介紹了mybatis-plus QueryWrapper 排序的坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01Java編程實(shí)現(xiàn)計(jì)算兩個(gè)日期的月份差實(shí)例代碼
這篇文章主要介紹了Java編程實(shí)現(xiàn)計(jì)算兩個(gè)日期的月份差實(shí)例代碼,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01SpringBoot項(xiàng)目使用mybatis-plus逆向自動(dòng)生成全套代碼
在JavaWeb工程中,每一個(gè)SSM新項(xiàng)目或者說是SpringBoot項(xiàng)目也好,都少不了model、controller、service、dao等層次的構(gòu)建。使用mybatis-plus逆向可以自動(dòng)生成,感興趣的可以了解一下2021-09-09