可能我们在测试控制层的代码都是启动服务器,在浏览器中输入URL,然后开始测试是否达到预期效果,发生错误的话,修改相关代码并重启服务器再次进行测试。分析一下这个过程,启动服务器-->打开浏览器-->输入URL-->等待返回结果-->修复bug-->重启服务器.....循环。

其中的缺点也挺明显的,在浏览器输入URL的地址,如果是GET请求还好,POST请求或者DELETE请求怎么办?只能借助其他工具,通过命令行编写curl语句,或者借助谷歌浏览器的postman插件,亦或者自己在代码中通过编写相应httpClient方法来实现测试,但是这几种方法都较为麻烦,而且测试用例并不能较好的保存。再说一个缺点,代码修改后,往往需要再次重启服务器,等待启动完毕才能接下来的测试过程。
如果tomcat服务器启动速度较慢,这将是一件非常痛苦的事情,测试验证也不方便,且依赖网络环境,这些原因导致测试起来很麻烦,而为了可以方便对Controller进行测试,且很好的保存和循环使用测试用例,则可以通过单元测试来解决,通过前面一篇文章,大家对于单元测试的便利性有了认识和体会,接下来通过引入MockMVC进行控制层的单元测试。
MockMvc实现了对Http请求的模拟,能够直接使用网络的形式,转换到Controller的调用,这样可以使得测试速度快、不依赖网络环境,而且提供了一套验证的工具,这样可以使得请求的验证统一而且很方便。

基于intellij,期间有红色报错请相应添加依赖,不强调

pom依赖:

          
            
org.springframework.boot
            
spring-boot-starter-test
            
test
        

controller:

     @GetMapping("/lskTest")    @ApiOperation(value="lskTest")    public String lskTest(@RequestParam String uid){        return uid;    }

alt+insert呼出快捷菜单

image.png

自动创建了test的包和class

在测试类前加上注解

@RunWith(SpringRunner.class)@SpringBootTest(classes = EurekaClientApplication.class)@WebAppConfiguration@ContextConfigurationpublic class RunControllerTest {

image.png

注意:我的springboottest的值为“EurekaClientApplication”

这是这个模块的main函数的名字:

image.png

然后加上

    @Autowired    private WebApplicationContext context;    private MockMvc mvc;

image.png

再加上一个before

    @Before    public void setUp() throws Exception {        mvc = MockMvcBuilders                .webAppContextSetup(context)                .build();    }

image.png

在真实需要测试的代码中加入

 @Test    public void testUserController() throws Exception {        RequestBuilder request = null;        //路径        request = get("/run/lskTest/")                //参数                .param("uid", "1wqeqweqweqweqweq")                //接受的数据类型                .accept(MediaType.APPLICATION_JSON);        mvc.perform(request)                //状态吗是否相等               .andExpect(status().isOk())                //得到的信息是否与特定字段匹配                .andExpect(content().string("1wqeqweqweqweqweq"))                //输出信息                .andDo(print());    }

image.png

image.png

测试成功!

那么数据库的回滚呢?

加上注解@Rollback

image.png

同时在方法中加上注解@Transactional

image.png

如果不希望回滚 将rollback改为false即可

image.png