springboot集成测试容器重启问题的处理

 更新时间:2021年11月03日 14:34:51   作者:micro_hz  
这篇文章主要介绍了springboot集成测试容器重启问题的处理,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

背景

spring boot test的项目中常用的测试框架, 最近在写集成测试的时候发现一个比较奇怪的问题,当我在运行多个测试用例的时候会偶尔重新启动整个容器上下文,由于后期业务逐渐复杂,大量的测试用例需要运行,这个问题直接导致回归测试的效率降低。

在这里插入图片描述

举个例子:

在这里插入图片描述

几个类:

@RunWith(SpringRunner.class)
@SpringBootTest(classes = TestApplication.class)
public class BaseApiTest {
    @Test
    public void init() {
    }
}
public class ApiTest1 extends BaseApiTest {
    @MockBean
    private Service service;
    @Test
    public void test1() {
        service.call();
    }
}
public class ApiTest2 extends BaseApiTest {
    @Autowired
    private Service service;
    @Test
    public void test2() {
        service.call();
    }
}
@SpringBootApplication
@Slf4j
public class TestApplication {
    public static void main(String[] args) {
        log.info("启动容器");
        new SpringApplication(TestApplication.class).run(args);
    }
}
@Component
public class Service {
    public void call() {
        System.out.println("service called");
    }
}

运行test包下所有测试:

在这里插入图片描述

发现容器重复启动了。

测试用例的运行流程

可以开启idea的线程堆栈跟踪,观察整个容器的启动路径

在这里插入图片描述

com.intellij.rt.junit是idea内部的实现,点击idea的运行单测会触发JunitStarter的main函数去启动,可以去GitHub找到源码:

在这里插入图片描述

做一些准备工作找到指定的runner就开始调用junit的包去执行编写的单测,junit为了灵活的扩展不同的测试运行环境,类似SPI机制动态获取Runner去运行单测。例如我的例子里指定了SpringRunner就是需要依赖Spring容器的一个实现,这样就让测试用例可以运行在Spring环境中。

在这里插入图片描述

junit的入口也支持在测例前后去插入一些操作,自己去实现RunnerListener即可。junit默认实现了监听器去记录测例的耗时,失败的数量等信息。

在这里插入图片描述

我指定的Runner是SpringRunner,其与SpringJUnit4ClassRunner并没什么区别,可以看其实现完全继承了SpringJUnit4ClassRunner的实现。

所以我们直接看SpringJUnit4ClassRunner的runner实现:

在这里插入图片描述

它首先判断了当前的环境是否需要忽略单测。如果忽略会在通知里得到通知。关于环境的指定控制可以参考注解@IfProfileValue,判断环境正确之后继续调用junit包父类ParentRunner的方法执行。

其定义了执行的基本的模板:

在这里插入图片描述

classBlock里面定义线程池执行和测例执行的一些before和after逻辑,里面的runChild是抽象方法,也是留给各个Runner实现的钩子。getFilteredChildren能够根据@Test注解拿到所有需要运行的用例方法,然后每个方法去调用具体的Runner运行。

在这里插入图片描述

其流程图如下

在这里插入图片描述

在这里插入图片描述

SpringJUnit4ClassRunner的运行每个方法会给每个测例方法进行一个封装成Statement。关键就在methodBlock方法,它实现了Spring boot对方法运行的封装

在这里插入图片描述

createTest会在测试的上下文里维护一个配置,然后会用通知机制一样去依次调用需要准备的东西,其中就包含spring容器的上下文。

在这里插入图片描述

在这里插入图片描述

其会执行injectDependencies,处理依赖的bean准备。TestContext是每个单测方法需要运行的上下文,在Spring boot的测试环境下,其维护了Spring的上下文,每个方法的执行都会去获取Spring的上下文

在这里插入图片描述

根据单测的相关信息文获取上spring的上下文,为了避免每次都去加载容器,TestContext会维护一个spring容器的缓存,CacheAwareContextLoaderDelegate

在这里插入图片描述

在这里插入图片描述

CacheAwareContextLoaderDelegate其内部的获取又是通过单测配置信息去ContextCache获取的,其内部是一个同步SynchronizedMap去保存的。

在这里插入图片描述

其内部实现看测例的配置信息去获取加载过的容器,如果没获取到就会触发重新加载新容器的流程,所以关键就是看key在Map中的获取逻辑,其底层是spring test自己实现一个的Map

在这里插入图片描述

可以看到其是基于HashMap的一个哈希结构,根据Jdk的源码,我们可以知道HashMap的key是根据hashCode与equals去比较key,那可以确定,要想复用同一个容器就得看Key值的hashCode和equals实现。接下来我们看MergedContextConfiguration源码.

在这里插入图片描述

在这里插入图片描述

可以发现其比较的值是:

 	 * @param testClass the test class for which the configuration was merged
	 * @param locations the merged context resource locations
	 * @param classes the merged annotated classes
	 * @param contextInitializerClasses the merged context initializer classes
	 * @param activeProfiles the merged active bean definition profiles
	 * @param propertySourceLocations the merged {@code PropertySource} locations
	 * @param propertySourceProperties the merged {@code PropertySource} properties
	 * @param contextCustomizers the context customizers
	 * @param contextLoader the resolved {@code ContextLoader}
	 * @param cacheAwareContextLoaderDelegate a cache-aware context loader
	 * delegate with which to retrieve the parent context
	 * @param parent the parent configuration or {@code null} if there is no parent

这些参数确定了能否共享SpringApplication,那两个测试类一个@Autowired,另一个使用@MockBean,肯定是改变这里面某个值,我们可以回溯这个MergedContextConfiguration是在什么时候被初始化的。这个还要追溯到idea的启动类,找到Runner的时候,SpringJUnit4ClassRunner的初始化的过程。

在每个测试类的运行都会唤起SpringJUnit4ClassRunner初始化,调用构造函数的时候会去加载测试类的上下文

Texrt

去创建这个TextContextManager

在这里插入图片描述

这里首先会根据被测试类的继承关系和注解的递归去找到固定包下面被注解@BootstrapWith修饰的类,因为是Spring boot test这里会根据@SpringBootTest 注解找到SpringBootTestContextBootstrapper类,找到这个引导类之后就会去初始化MergedContextConfiguration了。

在这里插入图片描述

引导类通过SPI机制加载到所有的Customizer,并根据需要DefinitionsParser,进行转换,保存在MergedContextConfiguration的一个字段,mock的一个属性会在转换的时候记录到,而非mock的contextCustomizers则不会记录。

注意这里提到的

在这里插入图片描述

在这里插入图片描述

两个类一个用mock的字段,一个用非mock的字段,两个MockitoContextCustomizer的definitions就不一样,因此无法共享上下文,因此需要重新启动一个Spring容器,并存放到CacheAwareContextLoaderDelegate,以便后面共享。

结论

分析源码的设计,发现应用了很多SPI与可扩展的设计,idea与junit的解耦,junit的抽象与模板定义与各个测试框架的扩展。

针对容器重启的角度,对于一个类来说,一定是共享一个spring上下文,但是不同的类可能由于注入的bean的方式不同导致无法共享spring上下文,所以导致重启会浪费掉一些时间,因此建议确定好mock的边界,对尽量多的测例共享一个容器视角可以提高单测效率,基于此可以设计多继承关系的单测结构,并把注入的bean向上共享,避免各个测试子类自己去注入出现不一致的情况。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • synchronized背后的monitor锁实现详解

    synchronized背后的monitor锁实现详解

    这篇文章主要为大家介绍了synchronized背后的monitor锁实现详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-09-09
  • java接入创蓝253短信验证码的实例讲解

    java接入创蓝253短信验证码的实例讲解

    下面小编就为大家分享一篇java接入创蓝253短信验证码的实例讲解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-01-01
  • Springboot集成任务调度实现过程

    Springboot集成任务调度实现过程

    这篇文章主要介绍了Springboot集成任务调度实现过程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • javaweb实现文件上传示例代码

    javaweb实现文件上传示例代码

    这篇文章主要为大家详细介绍了javaweb实现文件上传的示例代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-04-04
  • 线上Java程序占用CPU过高解决方案

    线上Java程序占用CPU过高解决方案

    这篇文章主要介绍了线上Java程序占用CPU过高解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-11-11
  • mybatis多对多关联实战教程(推荐)

    mybatis多对多关联实战教程(推荐)

    下面小编就为大家带来一篇mybatis多对多关联实战教程(推荐)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-10-10
  • Java初学者入门之继承和多态

    Java初学者入门之继承和多态

    Java 面向对象编程有三大特性:封装、继承、多态,学好继承和多态是面向对象开发语言中非常重要的一个环节,这篇文章主要给大家介绍了关于Java初学者入门之继承和多态的相关资料,需要的朋友可以参考下
    2021-07-07
  • Java提效神器Stream的一些冷门技巧汇总

    Java提效神器Stream的一些冷门技巧汇总

    这篇文章主要给大家介绍了关于Java提效神器Stream的一些冷门技巧,Stream是java对集合操作的优化,相较于迭代器,使用Stream的速度非常快,并且它支持并行方式处理集合中的数据,默认情况能充分利用cpu的资源,需要的朋友可以参考下
    2021-07-07
  • IDEA的Swing可视化插件JFormDesigner详解

    IDEA的Swing可视化插件JFormDesigner详解

    JFormDesigner是一个专业的软件应用程序,专门用于帮助您开发Java Swing用户界面,而无需具备编程技能。它可作为独立实用程序使用,也可以将其用作各种IDE的插件,本文给大家介绍idea Swing可视化插件,感兴趣的朋友一起看看吧
    2022-06-06
  • Mybatis或Mybatis-Plus框架的xml文件中特殊符号的使用详解

    Mybatis或Mybatis-Plus框架的xml文件中特殊符号的使用详解

    这篇文章主要介绍了Mybatis或Mybatis-Plus框架的xml文件中特殊符号的使用详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-11-11

最新评论