环境:SpringBoot2.6.12
被导入的类加入到Spring IOC容器中。
public class A { @Bean public B b() { return new B() ; }
在配置类中导入上面的类
@Import({A.class})@Configurationpublic class ImportConfig {}
这样A,B都被Spring IOC容器管理了。这里可以导入多个类。
注意:在Spring4.2之前的版本中被导入的类必须是配置类也就是类上有@Configuration注解,4.2后的版本随意一个普通类也可以。
测试
@Resourceprivate A a ;@Resourceprivate B b ; @Testpublic 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()方法中调用p()方法,那这里得到的Person对象和p()方法中定义的是同一个吗?
在上面的代码中一定不是同一个,输出如下:
p(), com.pack.main.importaware.ImportMain$Person@18dfcc1p(), com.pack.main.importaware.ImportMain$Person@19f7dcfd(), com.pack.main.importaware.ImportMain$Person@19f7dcf
分别是两个不同的对象,如何解决呢?我们只需要在A类上添加@Configuration即可,添加了该类后Spring容器首先会对当前的A这个类创建代理,当我们在A这个类中调用其它方法的时候会通过拦截器BeanMethodInterceptor进行拦截;在该拦截器中会根据你调用的方法来确定对应的beanName,然后在容器中查找是否有对应的Bean,如果有则直接返回,所以就确保了在当前类中你不管调用多少次其它@Bean方法都能保证是同一个对象。
通过实现ImportSelector接口
public class E { @Bean public G g() { return new G() ; }}public class F {}public class G {}
配置类
@Import({C.class, A.class})@Configurationpublic class ImportConfig {}
实现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数组(length = 0),但是不能返回null。实现ImportSelector该接口的这个本身是不会被注册为Bean的。
测试
@Resourceprivate E e ;@Resourceprivate F f ;@Resourceprivate G g ; @Testpublic void testImport() { System.out.println("e = " + e) ; System.out.println("f = " + f) ; System.out.println("g = " + g) ;}
图片
public class H implements ImportBeanDefinitionRegistrar { @Override public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) { GenericBeanDefinition beanDefinition = new GenericBeanDefinition() ; beanDefinition.setBeanClass(X.class) ; // 给X这个类配置属性name值 beanDefinition.getPropertyValues().addPropertyValue("name", "张三") ; // x 为当前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})@Configurationpublic class ImportConfig {}
测试:
图片
注意:实现ImportBeanDefinitionRegistrar该接口本身这个类是不会被注册为Bean的。
完毕!!!
本文链接://www.dmpip.com//www.dmpip.com/showinfo-26-11212-0.htmlSpring常用注解@Import的三种使用方式,你都了解了吗?
声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com