一般情况我们使用 @springbootapplication 注解来启动 springboot 项目
它其实只相当于 @configuration、@enableautoconfiguration、@componentscan(包含了两个filter)
@springbootapplicationpublic class frameworkunitrealtestapp { public static void main(string[] args) { springapplication.run(frameworkunitrealtestapp.class, args); }}
一般情况我们使用 @springboottest 和 @runwith(springrunner.class) 注解来启动 springboot 测试项目
@runwith(springrunner.class)@springboottestpublic class frameworkunitrealtestapp { @test public void test() {}}
这两个注解的区别的核心在于两个注解:@enableautoconfiguration、@componentscan(包含了两个filter)
@enableautoconfiguration 启动了所有的自动配置类
@componentscan(包含了两个filter):在扫描阶段过滤掉 @testcomponent 等专属于测试的类和过滤掉被 @configuration 注解的自动配置类(使得自动配置类不会在扫描阶段就被注册 beandefinition,因为 自动配置类的优先级应该是最低的)
可以看出 @springboottest 并没有启用任何自动配置类,所以就不需要加 autoconfigurationexcludefilter 了
springboot 通过引入 @test** 注解来在 测试环境下 引入不同的自动配置类!
详细的代码如下:添加了 typeexcludefilter 和 autoconfigurationexcludefilter 两个 excludefilter
作用:扫描包的时候过滤掉被这两个 filter 匹配的类!
@componentscan(excludefilters = { @filter(type = filtertype.custom, class = typeexcludefilter.class), @filter(type = filtertype.custom, class = autoconfigurationexcludefilter.class) })
4.1 typeexcludefilter 解析
主要移除测试相关的类
public class typeexcludefilter implements typefilter, beanfactoryaware { @overrdue toide public boolean match(metadatareader metadatareader, metadatareaderfactory metadatareaderfactory) throws ioexception { if (this.beanfactory instanceof listablebeanfactory && getclass() == typeexcludefilter.class) { collection<typeexcludefilter> delegates = ((listablebeanfactory) this.beanfactory) .getbeansoftype(typeexcludefilter.class).values(); for (typeexcludefilter delegate : delegates) { if (delegate.match(metadatareader, metadatareaderfactory)) { return true; } } } return fal; }}//delegate.match 走这个类的 match 方法class testtypeexcludefilter extends typeexcludefilter { private static final string[] class_annotations = { "org.junit.runner.runwith", "org.junit.jupiter.api.extension.extendwith" }; private static final string[] method_annotations = { "org.junit.test", "org.junit.platform.commons.annotation.testable" }; @override public boolean match(metadatareader metadatareader, metadatareaderfactory metadatareaderfactory) throws ioexception { //是否被 @testcomponent 及其父注解注释 if (istestconfiguration(metadatareader)) {return true;} //类上或类中方法上有没有 class_annotations、method_annotations 中的注解 if (istestclass(metadatareader)) {retur闭经治疗n true;} string enclosing = metadatareader.getclassmetadata().getenclosingclassname(); if (enclosing != null) { //递归内部类、父类 if (match(metadatareaderfactory.getmetadatareader(enclosing), metadatareaderfactory)) { return true; } } return fal; }}
4.2 autoconfigurationexcludefilter 解析
主要移除被 @configuration 修饰的 自动配置类
public class autoconfigurationexcludefilter implements typefilter, beanclassloaderaware { @override public boolean match(metadatareader metadatareader, metadatareaderfactory metadatareaderfactory) throws ioexception { //如果被 @configuration 注解,并且是 自动配置类就返回 true,即匹配成功 //注:被 @component 等注解并不匹配 return isconfiguration(metadatareader) &&am展示设计效果图p; isautoconfiguration(metadatareader); }}
作用:启用自动配置类
@autoconfigurationpackage//启用 autoconfigurationimportlector 配置类:扫描得到所有自动配置类@import(autoconfigurationimportlector.class)public @interface enableautoconfiguration { string enabled_override_property = "spring.boot.enableautoconfiguration"; //定义不启用的 自动配置类 class<?>[] exclude(广告词设计) default {}; //同上 string[] excludename() default {};}//这个注解主要是向容器中注册 autoconfigurationpackages.registrar 类用来存储自动配置包@import(autoconfigurationpackages.registrar.class)public @interface autoconfigurationpackage {}//关键:这个类继承了 deferredimportlector 接口,所以是到最后才解析的!!public class autoconfigurationimportlector implements deferredimportlector{ @override public string[] lectimports(annotationmetadata annotationmetadata) { if (!inabled(annotationmetadata)) { return no_imports; } autoconfigurationmetadata autoconfigurationmetadata = autoconfigurationmetadataloader .loadmetadata(this.beanclassloader); autoconfigurationentry autoconfigurationentry = getautoconfigurationentry( autoconfigurationmetadata, annotationmetadata); return stringutils.tostringarray(autoconfigurationentry.getconfigurations()); }}
spring boot 中文文档 对每个 @…test 注解导入的自动配置类做了详细的说明
springboottest 是测试使用类的注解,标志这个类是测试用例。
@target({elementtype.type})@retention(retentionpolicy.runtime)@documented@inherited@bootstrapwith(springboottestcontextbootstrapper.class)@extendwith({springextension.class})public @interface springboottest {
@target({elementtype.type})@retention(retentionpolicy.runtime)@documented@inherited@springbootconfiguration@enableautoconfiguration@componentscan( excludefilters = {@filter( type = filtertype.custom, class = {typeexcludefilter.class}), @filter( type = filtertype.custom, class = {autoconfigurationexcludefilter.class})})public @interface springbootapplication {
对比显示都是复合注解,并且前四个注解是一样的,后面区分bootstrapwith和extendw月开头的成语ith这两个是测试中包含的
bootstrapwith这个注解中有一个参数为springboottestcontextbootstrapper
这里面申明了一些程序运行所在包的路径,在去查看继承的顶级类可以追溯到testcontextbootstrapper 这个接口 :
从里面的方法可以看到是在运行的时候设置上下文 以及如何获取上下文,来提供测试启动的必须值。
可以看出这个实现了很多接口,来处理测试需要的各种通知处理,以及在测试接口时可以提前处理请求参数。
springbootapplication中的复合注解则是扫描一些包和配置。虽然测试也是项目启动的一种,可以看到里面实现还是有些区别的。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持www.887551.com。
本文发布于:2023-04-04 16:17:04,感谢您对本站的认可!
本文链接:https://www.wtabcd.cn/fanwen/zuowen/bd8ab7035ac6283d1ecde05f8bcb9096.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文word下载地址:关于@SpringBootApplication与@SpringBootTest的区别及用法.doc
本文 PDF 下载地址:关于@SpringBootApplication与@SpringBootTest的区别及用法.pdf
留言与评论(共有 0 条评论) |