存在这样一个场景: 

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

    这工作不在于使用编写代码,而在于每次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代码

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