存在这样一个场景: 

    当项目启动时间过长,又没办法缩短的时候,写单元测试就是一个十分耗时的工作,

    这工作不在于使用编写代码,而在于每次run junit test 都需要完整启动一次项目,白白浪费宝贵的生命。

    当由于某个字段没有赋值,或者某个简单判断写错,导致需要再等个3-5分钟启动 junit test,是否会想要执行一次san check?  

  于是乎:

    假若能使用controller来调用test类方法的话,那么在本地调试单元测试时,对于一些简单的代码修改,

    通过热部署,只需要重新进行一次url访问就可执行一个完整的单元测试,

    无需再次启动整个项目。

  正题:

    1. 如何在controller访问src/test ?

    2. 如何编写 ?

  如何在controller访问src/test

    maven项目的默认配置中, src/test目录是测试目录,不会被编译到jar中,也就是在controller调用时会报ClassNotFoundException

    解决办法最好的是在pom文件中修改maven默认的测试目录,将src/test 作为正常目录使用

    <build><!-- 将测试目录更改为其他目录 -->
<testSourceDirectory>src/main/test</testSourceDirectory>
</build>

  需要重新maven-update。update后重新将src/test use for building path

  完成后便可以成功启动项目,并可以正常访问。

  如何编写

    可以做一个参考:

controller,最主要内容在于使用controller时,junit的自动回滚可能不会生效,所以手动设置事务,手动触发回滚

@RestController
@RequestMapping(value = "/junit", produces = "application/json;charset=utf-8")
public class TestController { private final Logger logger = LoggerFactory.getLogger(TestController.class); @Autowired
private BizTest bizTest; @GetMapping("test")
@Transactional(rollbackFor = Exception.class)
public void startJunit() {
bizTest.insertTest();
// 手动开启事务回滚
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}
}

测试类,即可以通过其他类访问,也可以直接执行junit,增加@Componet 或者@Service 还可以通过spring注入方式调用

@Component
public class BizTest extends AbstractTest { @Autowired
private Biz biz; @Test
public void insertTest() {
// 测试业务
}
}

测试抽象父类,用于直接使用junit测试时的配置

@RunWith(SpringRunner.class)
@SpringBootTest(classes = WebApplication.class)
@Transactional
// @Rollback(false)
public abstract class AbstractTest { protected final Logger logger = LoggerFactory.getLogger(AbstractTest.class); protected void println(Object object) {
System.out.println(object);
}
}

改进: Controller 与Test类解耦

  直接在controller引用Test类,提交代码后容易引起打包错误,这里就需要对两者进行解耦。

  咱的解决办法如下:

     利用spring动态bean注册+类限定名来令两者解耦,并令Test类接受spring容器管理,代码如下:

@GetMapping("test")
@ResponseBody
public String startJunit() {
     // 测试主类限定名
String testImplName = "com.terra.test.JunitTester";
     // 若bean未注册,则注册,SpringConetxtUtil可参考其余地方的Spring动态注册bean
if (SpringContextUtil.getBean(testImplName) == null) {
registerBean(testImplName);
}
try {
IJunitTest junitTester = SpringContextUtil.getBean(testImplName);
junitTester.startTest();
} catch (Exception e) {
if (e.getClass() != UnexpectedRollbackException.class) {
return e.getMessage();
}
}
logger.info("执行单元测试成功");
return "操作成功, 执行完成时间:" + DateUtil.getNowDateTime();
}
  // 注册bean
private void registerBean(String className) {
DefaultListableBeanFactory beanFactory =
(DefaultListableBeanFactory) SpringContextUtil.getApplicationContext().getAutowireCapableBeanFactory();
if (beanFactory.getBean(className) == null) {
BeanDefinition beanDefinition = new GenericBeanDefinition();
beanDefinition.setBeanClassName(className);
beanFactory.registerBeanDefinition(className, beanDefinition);
}
}
  // 单元测试主类接口
public static interface IJunitTest {
void startTest();
}

测试主类代码:

@Component
public class JunitTester extends AbstractTest implements IJunitTest { @Autowired
private BizTest1 bizTest1;
@Autowired
private BizTest2 bizTest2; @Transactional(rollbackFor = Exception.class)
@Override
public void startTest() {
// 手动开启事务回滚
bizTest1.before();
bizTest1.test();
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}
}

这样做的好处是,环境上的代码没有src/test也不会报错,而本地只需要进行注释和去除注释@RequestMapping,便无需要再修改controller代码

所有单元测试代码操作便可在测试主类及其测试类中进行,可喜可贺。

【maven】【spring boot】【单元测试】 使用controller 执行单元测试类的更多相关文章

  1. Spring Boot Web 开发@Controller @RestController 使用教程

    在 Spring Boot 中,@Controller 注解是专门用于处理 Http 请求处理的,是以 MVC 为核心的设计思想的控制层.@RestController 则是 @Controller ...

  2. Spring Boot(十二)单元测试JUnit

    一.介绍 JUnit是一款优秀的开源Java单元测试框架,也是目前使用率最高最流行的测试框架,开发工具Eclipse和IDEA对JUnit都有很好的支持,JUnit主要用于白盒测试和回归测试. 白盒测 ...

  3. 如何对Spring MVC中的Controller进行单元测试

    对Controller进行单元测试是Spring框架原生就支持的能力,它可以模拟HTTP客户端发起对服务地址的请求,可以不用借助于诸如Postman这样的外部工具就能完成对接口的测试. 具体来讲,是由 ...

  4. Spring Boot 系列(二)单元测试&网络请求

    实际开发中,Junit单元测试是必不可少的.在spring-boot 中可以通过测试模块(spring-boot-starter-test)快速使用单元测试功能. 开始 本示例在 spring boo ...

  5. Spring boot 项目导出可执行jar

    配置文件中添加插件 <plugin> <groupId>org.springframework.boot</groupId> <artifactId>s ...

  6. Eclipse完成Maven + Spring Boot + Mybatis + jsp

    Spring Boot 完成WEB项目开发 开发工具:eclipse 框架:Maven:Spring Boot:Mybatis 界面:jsp:javascript:css 前言: 在SpringBoo ...

  7. 将 Spring boot 项目打成可执行Jar包,及相关注意事项(main-class、缺少 xsd、重复打包依赖)

    最近在看 spring boot 的东西,觉得很方便,很好用.对于一个简单的REST服务,都不要自己部署Tomcat了,直接在 IDE 里 run 一个包含 main 函数的主类就可以了. 但是,转念 ...

  8. spring boot mybatis 打成可执行jar包后启动UnsatisfiedDependencyException异常

    我的spring boot + mybatis项目在idea里面执行正常,但发布测试环境打成可执行jar包后就启动失败,提示错误如下: [ ERROR] [2018-08-30 17:23:48] o ...

  9. Spring boot进阶-配置Controller、interceptor...

    1.配置SpringBootApplication(对spring boot来说这是最基本) package io.github.syske.springboot31; import org.spri ...

随机推荐

  1. Angular 表单验证类库 ngx-validator 1.0 正式发布

    背景介绍 之前写了一篇 <如何优雅的使用 Angular 表单验证>,结尾处介绍了统一验证反馈的类库  ngx-validator  ,由于这段时间一直在新模块做微前端以及相关业务组件库, ...

  2. LinqToSQL4

    Join和GroupJoin的区别 List<Atable> ainfo = new List<Atable> { new Atable{ AId=1, AName=" ...

  3. javaIO——PushbackReader

    1. 注释解释: A character-stream reader that allows characters to be pushed back into the stream. 一个允许字符被 ...

  4. scala的泛型浅析

    1. scala泛型浅析 package com.dtspark.scala.basics /** * 1,scala的类和方法.函数都可以是泛型. * * 2,关于对类型边界的限定分为上边界和下边界 ...

  5. 华为机试题:仿LISP

    package com.nowcoder.huawei; import java.util.*; public class LISP { // 只通过80% // (+ (* 2 3) (^ 4)) ...

  6. boost交叉编译

    运行bootstrap.sh # ./bootstrap.sh 生成b2.bjam和project-config.jam文件 修改project-config.jam using gcc : arm ...

  7. SpringCloud之Hystrix容错保护原理及配置

    1 什么是灾难性雪崩效应? 如下图的过程所示,灾难性雪崩形成原因就大致如此: 造成灾难性雪崩效应的原因,可以简单归结为下述三种: 服务提供者不可用.如:硬件故障.程序BUG.缓存击穿.并发请求量过大等 ...

  8. Emeditor代码编辑器常见的正则表达式总结

    Emeditor 目前来说是我个人感觉非常不错的一款记事本软件, 其中查找替换功能由于支持正则表达式而显得非常强大. <tr[^>]*> 匹配:<tr xxxxxxxxxxxx ...

  9. JavaMaven【八、pom.xml】

    简介: 重点学习: 1.dependency-scope 依赖范围 compile 编译 默认,对编译.测试.运行都有效 provided 编译和测试时有效 runtime 测试和运行时有效 test ...

  10. springboot Invalid character found in the request target. The valid characters are defined in RFC 7230 and RFC 3986

    报错如下: 在请求目标中发现无效字符.有效字符在RFC 7230和RFC 3986中定义. 原因是Tomcat在 7.0.73, 8.0.39, 8.5.7 版本后,添加了对于http头的验证. 就是 ...