Spring常用注解@Import的三種使用方式,你都了解了嗎?
環(huán)境:SpringBoot2.6.12
三種方法說明
- 直接導(dǎo)入普通數(shù)組的方式
- 導(dǎo)入ImportSelector類方式
- 導(dǎo)入ImportBeanDefinitionRegistrar方式
被導(dǎo)入的類加入到Spring IOC容器中。
直接導(dǎo)入普通數(shù)組的方式
public class A {
@Bean
public B b() {
return new B() ;
}
在配置類中導(dǎo)入上面的類
@Import({A.class})
@Configuration
public class ImportConfig {
}
這樣A,B都被Spring IOC容器管理了。這里可以導(dǎo)入多個(gè)類。
注意:在Spring4.2之前的版本中被導(dǎo)入的類必須是配置類也就是類上有@Configuration注解,4.2后的版本隨意一個(gè)普通類也可以。
測試
@Resource
private A a ;
@Resource
private B b ;
@Test
public void testImport() {
System.out.println("a = " + a) ;
System.out.println("b = " + b) ;
}
圖片
沒問題,能正常的輸出。
注意,請看下面的示例:
public class ImportMain {
static class Person{
}
static class A{
@Bean
public Person p() {
Person person = new Person();
System.out.println("p(), " + person) ;
return person ;
}
@Bean
public Date d() {
Person p = p() ;
System.out.println("d(), " + p) ;
return new Date() ;
}
}
@Import({A.class})
@Configuration
static class AppConfig {
}
}
在上面的示例中在d()方法中調(diào)用p()方法,那這里得到的Person對象和p()方法中定義的是同一個(gè)嗎?
在上面的代碼中一定不是同一個(gè),輸出如下:
p(), com.pack.main.importaware.ImportMain$Person@18dfcc1
p(), com.pack.main.importaware.ImportMain$Person@19f7dcf
d(), com.pack.main.importaware.ImportMain$Person@19f7dcf
分別是兩個(gè)不同的對象,如何解決呢?我們只需要在A類上添加@Configuration即可,添加了該類后Spring容器首先會對當(dāng)前的A這個(gè)類創(chuàng)建代理,當(dāng)我們在A這個(gè)類中調(diào)用其它方法的時(shí)候會通過攔截器BeanMethodInterceptor進(jìn)行攔截;在該攔截器中會根據(jù)你調(diào)用的方法來確定對應(yīng)的beanName,然后在容器中查找是否有對應(yīng)的Bean,如果有則直接返回,所以就確保了在當(dāng)前類中你不管調(diào)用多少次其它@Bean方法都能保證是同一個(gè)對象。
導(dǎo)入ImportSelector類方式
通過實(shí)現(xiàn)ImportSelector接口
public class E {
@Bean
public G g() {
return new G() ;
}
}
public class F {
}
public class G {
}
配置類
@Import({C.class, A.class})
@Configuration
public class ImportConfig {
}
實(shí)現(xiàn)ImportSelector接口
public class C implements ImportSelector {
@Override
public String[] selectImports(AnnotationMetadata importingClassMetadata) {
return new String[] {"com.pack.import_annotation.E", "com.pack.import_annotation.F"};
}
}
這里的返回值中必須是完整的包名+類名
注意:這里可以返回空的String數(shù)組(length = 0),但是不能返回null。實(shí)現(xiàn)ImportSelector該接口的這個(gè)本身是不會被注冊為Bean的。
測試
@Resource
private E e ;
@Resource
private F f ;
@Resource
private G g ;
@Test
public void testImport() {
System.out.println("e = " + e) ;
System.out.println("f = " + f) ;
System.out.println("g = " + g) ;
}
圖片
導(dǎo)入ImportBeanDefinitionRegistrar方式
public class H implements ImportBeanDefinitionRegistrar {
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
GenericBeanDefinition beanDefinition = new GenericBeanDefinition() ;
beanDefinition.setBeanClass(X.class) ;
// 給X這個(gè)類配置屬性name值
beanDefinition.getPropertyValues().addPropertyValue("name", "張三") ;
// x 為當(dāng)前X類在Spring容器中的beanName
registry.registerBeanDefinition("x", beanDefinition) ;
}
}
配置類
public class X {
private String name ;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
@Import({C.class, A.class, H.class})
@Configuration
public class ImportConfig {
}
測試:
圖片
注意:實(shí)現(xiàn)ImportBeanDefinitionRegistrar該接口本身這個(gè)類是不會被注冊為Bean的。
完畢?。?!