1. Spring Auto Import(基于 @EnableAutoConfiguration)
@SpringBootApplication
注解中有一個@EnableAutoConfiguration
注解砸琅,它負(fù)責(zé)啟用了springboot
的自動裝配,實現(xiàn)自動裝配前會準(zhǔn)備一個預(yù)定義的清單文件(AutoConfiguration.imports/spring.factories)
然后根據(jù)類路徑生成候選自動配置類,在經(jīng)過條件判斷扁远,最后這些配置類會被實例化并注冊到ioc容器中
AutoConfiguration.imports 文件的生成
SpringBoot3中提供了@AutoConfiguration注解敬辣,在借助構(gòu)建工具(如maven)及
spring提供的spring-boot-maven-plugin
進(jìn)行構(gòu)建時称勋,會識別@AutoConfiguration注解將這些帶有注解的類全限定名路徑存儲在AutoConfiguration.imports 文件耀鸦,這些類全部成為候選自動配置類捶枢。
讀取及篩選配置類的流程
這個方法就是最后將列表中所有的配置類注冊到ioc容器上火邓,
這里的條件判斷指的是配置類上的@Conditional 系列注解,方法最后,調(diào)用 getConfigurations() 方法獲取候選配置類的列表铲咨,并將其轉(zhuǎn)換為字符串?dāng)?shù)組返回躲胳。
實現(xiàn)裝配
最后裝配是通過EnableAutoConfiguration.class
上的@Import(AutoConfigurationImportSelector.class)
實現(xiàn)的。
Import
注解通過Spring 的 ImportSelector 和 ImportBeanDefinitionRegistrar 接口提供了兩種方式來實現(xiàn)更復(fù)雜的導(dǎo)入邏輯纤勒。
2.@Import配合條件注解
@Import注解最后指向返回的都是配置類或是配置類的全限定名坯苹,所以說重構(gòu)selectImports方法或者是配置類上的@Conditional 系列注解,它都是在裝配到ioc容器前的篩選的一步摇天,通過@Configuration標(biāo)識類將最后通過@Conditional 系列注解的@bean裝配到ioc容器粹湃。
標(biāo)注: 這里的config配置類是在ioc容器中的,所以通過@Conditional 系列注解進(jìn)行裝配是動態(tài)的闸翅。所以我說
都是在裝配到ioc容器前的篩選的這個說法并不準(zhǔn)確再芋。
通過@Import配合條件注解可以指向你的自定義配置類,在你的配置類可以加上@Conditional 系列注解使其在什么情況下裝配你的bean坚冀。
最后像這樣济赎,你完成了基于你邏輯的自定義自動裝配。
3.SPI
jdk的SPI機(jī)制
Message 接口:提供服務(wù)的接口
public interface Message {
String getMessage();
}
HelloMessageService ,HiMessageService 實現(xiàn)服務(wù)的實現(xiàn)類
public class HelloMessageService implements Message{
@Override
public String getMessage() {
return "Hello Message Service!";
}
}
public class HiMessageService implements Message{
@Override
public String getMessage() {
return "Hi,Message Service";
}
}
Java的類加載器從META-INF/services/
目錄下加載Message
服務(wù)文件,文件名命名規(guī)則全限定類名例如:
# 文件位置:src/main/resources/META-INF/services/com.example.service.PaymentService
com.example.service.impl.HelloMessageService # 此處是文件內(nèi)容
com.example.service.impl.HiMessageService
package cn.Jason;
import cn.Jason.JdkSPI.Message;
import java.util.ServiceLoader;
public class MyApp {
public static void main(String[] args) {
ServiceLoader<Message> loader = ServiceLoader.load(Message.class);
for (Message message : loader) {
System.out.println(message.getMessage());
}
}
}
#執(zhí)行結(jié)果
Hello Message Service!
Hi,Message Service
這里注意一點记某,loader
迭代到某類時那個實現(xiàn)類才實例化司训。
SpringSPI
因為spring 的實例化在容器內(nèi)進(jìn)行,所以這里只需完成服務(wù)發(fā)現(xiàn)和對應(yīng)的實現(xiàn)配置類液南,將他們注冊到容器就完成了自動裝配功能壳猜。
在springboot2.7
以下版本META-INF/spring.factories
的文件中這樣的鍵值對
cn.Jason.JdkSPI.Message =\
com.example.impl.HelloMessageService ,\
com.example.impl.HiMessageService
對比jdk的服務(wù)文件,spring
將服務(wù)的全限定類名作為鍵滑凉,將實現(xiàn)配置類作為值统扳。多個配置用,
分隔畅姊。
Springboot2與SpringBoot3處理區(qū)別
springboot2
在springboot2.7
以下版本spring.factories
文件中以EnableAutoConfiguration
為鍵咒钟,它會映射到一組自動配置類的列表(全限定類名)。
springboot2.7
以下版本讀取到文件之后的邏輯就是通過SpringFactoriesLoader
根據(jù)@EnableAutoConfiguration
的全限定類名作為鍵若未,查找其對應(yīng)的實現(xiàn)類列表朱嘴。
springboot3
AutoConfiguration.imports
文件的內(nèi)容則不是鍵值對格式,此文件中的每一行都是一個自動配置類的全限定類名(沒有鍵值對)粗合。
由于不是鍵值對格式萍嬉,所以直接將autoconfigure-imports
文件中的類名匯總成一個列表。
最后
這里也是對應(yīng)了我最開始說的AutoConfiguration.imports 文件
的生成,將候選配置類列表進(jìn)行篩選隙疚,再通過重構(gòu)selectImports方法
將配置類都注冊到ioc容器
中壤追,然后進(jìn)行通過@Conditional 系列注解
進(jìn)行條件判斷,實現(xiàn)動態(tài)裝配bean
供屉。