Spring注解之@Import的簡(jiǎn)單介紹
@Import可以導(dǎo)入以下幾種種類(lèi):
普通類(lèi)
實(shí)現(xiàn)ImportSelector接口的類(lèi)
實(shí)現(xiàn)DeferredImportSelector接口的類(lèi)
實(shí)現(xiàn)ImportBeanDefinitionRegistrar接口的類(lèi)
普通類(lèi)
被導(dǎo)入的類(lèi)會(huì)被容器注冊(cè)成一個(gè)Bean,可以被依賴(lài)注入使用?!?.2 版本之前只可以導(dǎo)入配置類(lèi);4.2版本之后也可以導(dǎo)入普通類(lèi),導(dǎo)入的類(lèi)會(huì)被當(dāng)作配置類(lèi)】
@Import注冊(cè)一個(gè)類(lèi)時(shí),這個(gè)配置類(lèi)不應(yīng)該被@Component或者@Configuration注解標(biāo)記。Spring中會(huì)將所有的bean class封裝成一個(gè)ConfigurationClass,并且此后會(huì)判斷被封裝的bean class是否是由其他類(lèi)導(dǎo)入的.
@Configuration
<strong>@Import(OtherBean.</strong><strong>class)
</strong>public class SpringConfig { }public class OtherBean { }ImportSelector實(shí)現(xiàn)類(lèi)
實(shí)現(xiàn)類(lèi)不會(huì)被注冊(cè)成Bean,接口方法的返回值會(huì)被注冊(cè)成Bean。【BeanName是全類(lèi)名】
@Configuration
<strong>@Import(MyImportSelector.</strong><strong>class)
</strong>public class SpringConfig { }public class MyImportSelector implements<strong> ImportSelector</strong> {
@Override
public String[] selectImports(AnnotationMetadata importingClassMetadata) {
return new String[]{OtherBean.class.getName()};
}
}DeferredImportSelector實(shí)現(xiàn)類(lèi)
DeferredImportSelector是ImportSelector的子接口, 所以它們的實(shí)現(xiàn)方式一樣,只是Spring的處理方式不同。DeferredImportSelector和SpringBoot中自動(dòng)導(dǎo)入配置文件的延遲導(dǎo)入有關(guān)。
@Configuration
<strong>@Import(MyDeferredImportSelector.</strong><strong>class)
</strong>public class SpringConfig { }public class MyDeferredImportSelector implements<strong> DeferredImportSelector</strong> {
@Override
public String[] selectImports(AnnotationMetadata importingClassMetadata) {
return new String[]{OtherBean.class.getName()};
}
}ImportBeanDefinitionRegistrar實(shí)現(xiàn)類(lèi)
實(shí)現(xiàn)類(lèi)不會(huì)被注冊(cè)為bean,但是會(huì)回調(diào)其接口方法,由開(kāi)發(fā)者通過(guò)Spring api手動(dòng)向Spring容器注冊(cè)bean?!绢?lèi)似于BeanFactoryPostRegister】
@Configuration
<strong>@Import(MyImportBeanDefinitionRegistrar.</strong><strong>class)
</strong>public class SpringConfig { }public class MyImportBeanDefinitionRegistrar implements<strong> ImportBeanDefinitionRegistrar</strong> {
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
BeanDefinition beanDefinition = new RootBeanDefinition();
String beanName = StringUtils.uncapitalize(OtherBean.class.getSimpleName());
beanDefinition.setBeanClassName(OtherBean.class.getName());
registry.registerBeanDefinition(beanName,beanDefinition);
}
}附:@Import相關(guān)源碼解析
加載解析@Import注解位于BeanFactoryPostProcessor處理的時(shí)候:
AbstractApplicationContext的refresh方法
-> invokeBeanFactoryPostProcessors(beanFactory);
-> PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors());
-> registryProcessor.postProcessBeanDefinitionRegistry(registry);
這里的registryProcessor,我們指ConfigurationClassPostProcessor
ConfigurationClassPostProcessor.postProcessBeanDefinitionRegistry(registry)
-> processConfigBeanDefinitions(registry):
public void processConfigBeanDefinitions(BeanDefinitionRegistry registry) {
//省略一些配置檢查與設(shè)置的邏輯
//根據(jù)@Order注解,排序所有的@Configuration類(lèi)
configCandidates.sort((bd1, bd2) -> {
int i1 = ConfigurationClassUtils.getOrder(bd1.getBeanDefinition());
int i2 = ConfigurationClassUtils.getOrder(bd2.getBeanDefinition());
return Integer.compare(i1, i2);
});
// 創(chuàng)建ConfigurationClassParser解析@Configuration類(lèi)
ConfigurationClassParser parser = new ConfigurationClassParser(
this.metadataReaderFactory, this.problemReporter, this.environment,
this.resourceLoader, this.componentScanBeanNameGenerator, registry);
//剩余沒(méi)有解析的@Configuration類(lèi)
Set<BeanDefinitionHolder> candidates = new LinkedHashSet<>(configCandidates);
//已經(jīng)解析的@Configuration類(lèi)
Set<ConfigurationClass> alreadyParsed = new HashSet<>(configCandidates.size());
do {
//解析
parser.parse(candidates);
parser.validate();
Set<ConfigurationClass> configClasses = new LinkedHashSet<>(parser.getConfigurationClasses());
configClasses.removeAll(alreadyParsed);
// 生成類(lèi)定義讀取器讀取類(lèi)定義
if (this.reader == null) {
this.reader = new ConfigurationClassBeanDefinitionReader(
registry, this.sourceExtractor, this.resourceLoader, this.environment,
this.importBeanNameGenerator, parser.getImportRegistry());
}
this.reader.loadBeanDefinitions(configClasses);
alreadyParsed.addAll(configClasses);
candidates.clear();
if (registry.getBeanDefinitionCount() > candidateNames.length) {
//省略檢查是否有其他需要加載的配置的邏輯
}
}
while (!candidates.isEmpty());
//省略后續(xù)清理邏輯
}
其中parser.parse(candidates)的邏輯主要由org.springframework.context.annotation.ConfigurationClassParser實(shí)現(xiàn),功能是加載@Import注解還有即系@Import注解。reader.loadBeanDefinitions(configClasses);的邏輯主要由org.springframework.context.annotation.ConfigurationClassBeanDefinitionReader的loadBeanDefinitionsForConfigurationClass方法實(shí)現(xiàn),功能是將上面解析的配置轉(zhuǎn)換為BeanDefinition就是Bean定義。
總結(jié)
到此這篇關(guān)于Spring注解之@Import的文章就介紹到這了,更多相關(guān)Spring注解@Import內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
vue+springboot+shiro+jwt實(shí)現(xiàn)登錄功能
這篇文章主要介紹了vue+springboot+shiro+jwt實(shí)現(xiàn)登錄功能,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04
Java Quartz觸發(fā)器CronTriggerBean配置用法詳解
這篇文章主要介紹了Java Quartz觸發(fā)器CronTriggerBean配置用法詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08
Spring boot集成redis lettuce代碼實(shí)例
這篇文章主要介紹了Spring boot集成redis lettuce代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04
Springboot mybais配置多數(shù)據(jù)源過(guò)程解析
這篇文章主要介紹了Springboot+mybais配置多數(shù)據(jù)源過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03
Java中高效的對(duì)象映射庫(kù)Orika的用法詳解
Orika是一個(gè)高效的Java對(duì)象映射庫(kù),專(zhuān)門(mén)用于在Java應(yīng)用程序中簡(jiǎn)化對(duì)象之間的轉(zhuǎn)換,下面就跟隨小編一起來(lái)深入了解下Orika的具體使用吧2024-11-11
java面向?qū)ο笤O(shè)計(jì)原則之接口隔離原則示例詳解
這篇文章主要為大家介紹了java面向?qū)ο笤O(shè)計(jì)原則之接口隔離原則的示例詳解,有需要的朋友可以借鑒參考下希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪2021-10-10

