存在这样一个场景:
当项目启动时间过长,又没办法缩短的时候,写单元测试就是一个十分耗时的工作,
这工作不在于使用编写代码,而在于每次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代码
所有单元测试代码操作便可在测试主类及其测试类中进行,可喜可贺。