SpringBoot自動裝配原理小結(jié)
約定優(yōu)于配置(Convention Over Configuration)是一種軟件設(shè)計范式,目的在于減少配置的數(shù)量或者降低理解難度,從而提升開發(fā)效率。
先總結(jié)一下結(jié)論:
springboot通過spring.factories能把main方法所在類路徑以外的bean自動加載,其目的就是為了幫助自動配置bean,減輕配置量
springboot autoconfig的一些實驗一個springboot工程,springbootautoconfig.test.config這個包和啟動類的包不再同一個路徑下,那么這個包下的注解等應(yīng)該不會生效,bean也無法托管給spring管理
@Configuration//開啟配置@EnableConfigurationProperties(HelloProperties.class)//開啟使用映射實體對象@ConditionalOnClass(TestHello.class)//存在TestHello時初始化該配置類@ConditionalOnProperty//存在對應(yīng)配置信息時初始化該配置類(prefix = 'zxp.hello',//存在配置前綴zxp.hellovalue = 'flag')public class HelloAutoConfiguration { @Autowired private HelloProperties helloProperties; @Bean//創(chuàng)建HelloService實體bean @ConditionalOnMissingBean(TestHello.class)//缺失HelloService實體bean時,初始化HelloService并添加到SpringIoc public TestHello helloService() {System.out.println('>>>The TestHello Not Found,Execute Create New Bean.');TestHello testHello = new TestHello(helloProperties.getName(),helloProperties.getFlag());return testHello; }}
@ConfigurationProperties(prefix = 'zxp.hello')@Datapublic class HelloProperties { private String name; private String flag;}
public class TestHello { String name; String flag; public TestHello(String name, String flag) {this.name = name;this.flag = flag; } public String print(){String msg = 'name is '+name + ' ' + 'flag is '+flag;System.out.println(msg);return msg; }}
在resources下創(chuàng)建META-INF路徑,并創(chuàng)建spring.factories文件
#配置自定義Starter的自動化配置org.springframework.boot.autoconfigure.EnableAutoConfiguration=springbootautoconfig.test.config.HelloAutoConfiguration
再試啟動又報錯了
- Bean method ’helloService’ not loaded because @ConditionalOnProperty (zxp.hello) did not find property ’flag’
原因是,如果沒有配置zxp.hello.flag,怎會報錯
@ConditionalOnProperty//存在對應(yīng)配置信息時初始化該配置類(prefix = 'zxp.hello',//存在配置前綴hello value = 'flag'//開啟)
在application.properties中添加
zxp.hello.flag=2
成功了,訪問controller
name is null flag is 1SpringBoot autoconfig部分注解說明
@ConditionalOnXxx 可以根據(jù)條件來決定是否執(zhí)行自動配置
@ConditionalOnBean:當(dāng)SpringIoc容器內(nèi)存在指定Bean的條件@ConditionalOnSingleCandidate:當(dāng)指定Bean在SpringIoc容器內(nèi)只有一個,或者雖然有多個但是指定首選的Bean@ConditionalOnMissingBean:當(dāng)SpringIoc容器內(nèi)不存在指定Bean的條件@ConditionalOnClass:當(dāng)SpringIoc容器內(nèi)存在指定Class的條件@ConditionalOnMissingClass:當(dāng)SpringIoc容器內(nèi)不存在指定Class的條件@ConditionalOnExpression:基于SpEL表達(dá)式作為判斷條件@ConditionalOnJava:基于JVM版本作為判斷條件@ConditionalOnJndi:在JNDI存在時查找指定的位置@ConditionalOnResource:類路徑是否有指定的值@ConditionalOnProperty:指定的屬性是否有指定的值@ConditionalOnNotWebApplication:當(dāng)前項目不是Web項目的條件@ConditionalOnWebApplication:當(dāng)前項目是Web項目的條件@AutoConfigureBefore@AutoConfigureAfter@AutoConfigureOrder
public @interface ConditionalOnProperty { String[] value() default {}; //數(shù)組,獲取對應(yīng)property名稱的值,與name不可同時使用 String prefix() default '';//property名稱的前綴,可有可無 String[] name() default {};//數(shù)組,property完整名稱或部分名稱(可與prefix組合使用,組成完整的property名稱),與value不可同時使用 String havingValue() default '';//可與name組合使用,比較獲取到的屬性值與havingValue給定的值是否相同,相同才加載配置 boolean matchIfMissing() default false;//缺少該property時是否可以加載。如果為true,沒有該property也會正常加載;反之報錯 boolean relaxedNames() default true;//是否可以松散匹配,至今不知道怎么使用的 } SpringBoot autoconfig原理
springboot啟動類注解@SpringBootApplication引入@EnableAutoConfiguration又引入@Import(AutoConfigurationImportSelector.class)
AutoConfigurationImportSelector類中調(diào)用SpringFactoriesLoader.loadFactoryNames 方法掃描了所有JAR包的META-INF/spring.factories,如下代碼:
protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) { List<String> configurations = SpringFactoriesLoader.loadFactoryNames(getSpringFactoriesLoaderFactoryClass(), getBeanClassLoader()); Assert.notEmpty(configurations, 'No auto configuration classes found in META-INF/spring.factories. If you ' + 'are using a custom packaging, make sure that file is correct.'); return configurations;}public static final String FACTORIES_RESOURCE_LOCATION = 'META-INF/spring.factories';public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) { …… Enumeration<URL> urls = (classLoader != null ? classLoader.getResources(FACTORIES_RESOURCE_LOCATION) : ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION)); result = new LinkedMultiValueMap<>(); while (urls.hasMoreElements()) {
spring-boot-autoconfigure包內(nèi)的spring.factories文件內(nèi)容
……work.boot.autoconfigure.EnableAutoConfiguration=org.springframework.boot.autoconfigure.admin.SpringApplicationAdminJmxAutoConfiguration,org.springframework.boot.autoconfigure.aop.AopAutoConfiguration,org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration,……
包含了所有spring為其增加的自動配置的bean配置,這些bean在滿足條件后會被加載到spring上下文中,從而實現(xiàn)了自動配置
eg:
@Configuration(proxyBeanMethods = false)@ConditionalOnClass({ RabbitTemplate.class, Channel.class })@EnableConfigurationProperties(RabbitProperties.class)@Import(RabbitAnnotationDrivenConfiguration.class)public class RabbitAutoConfiguration {
會發(fā)現(xiàn)RabbitTemplate會報錯,為什么這里不存在的類卻不報錯呢? 1、這個jar編譯時這個類是有的,保證編譯能過 2、看下ConditionalOnClass注解的注釋
The classes that must be present. Since this annotation is parsed by loading class bytecode, > it is safe to specify classes here that may ultimately not be on the classpath, only if this annotation is directly on the affected component and not if this annotation is used as a composed, meta-annotation. In order to use this annotation as a meta-annotation, only use the name attribute. Returns: the classes that must be present
必須出現(xiàn)的類。由于此注釋是通過加載類字節(jié)碼來解析的,因此在此處指定最終可能不在類路徑上的類是安全的,前提是此注釋直接位于受影響的組件上,而不是將此注釋用作組合的元注釋。要將此注釋用作元注釋,請僅使用name屬性。
starterstarter就是整理了依賴的maven配置,主要指maven相關(guān)依賴配置到單獨的一個工程以避免引入過多的maven配置
以上就是SpringBoot自動裝配原理詳解的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot自動裝配原理的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. Java 生成帶Logo和文字的二維碼2. XHTML 1.0:標(biāo)記新的開端3. 基于javaweb+jsp實現(xiàn)企業(yè)財務(wù)記賬管理系統(tǒng)4. AspNetCore&MassTransit Courier實現(xiàn)分布式事務(wù)的詳細(xì)過程5. 怎樣才能用js生成xmldom對象,并且在firefox中也實現(xiàn)xml數(shù)據(jù)島?6. 低版本IE正常運行HTML5+CSS3網(wǎng)站的3種解決方案7. xml中的空格之完全解說8. Android自定義控件實現(xiàn)方向盤效果9. Javaweb工程運行報錯HTTP Status 404解決辦法10. asp讀取xml文件和記數(shù)
