1.在`Component`标记的类下导入第三方的类2、注入实现了`importSelector`接口的类3、注入实现了`importBeanDefinitionRegistrar`接口的类
我们通过@import注解的注释可以知道,我们可以在被Component标记的类下导入其他的类和实现importSelector、importBeanDefinitionRegistrar接口的类
1.在Component标记的类下导入第三方的类
我们知道@import注解最最要的作用是用来批量导入第三方Jar包里面的类,因为第三方Jar包,并没有我们被我们Spring的注解所标记,对于少数的类我们可以通过@Bean注解导入IOC容器,想要批量导入时就可以用@import注解
栗子:
@Configuration@import({Apple.class,Orange.class, Banana.class})public class myConfig {}
测试一下:
@Testpublic void test(){ ApplicationContext context = context = new AnnotationConfigApplicationContext(myConfig.class); String[] beanDefinitionNames = context.getBeanDefinitionNames(); Arrays.asList(beanDefinitionNames).forEach(System.out::println);}
打印结果:
org.springframework.context.annotation.internalConfigurationAnnotationProcessororg.springframework.context.annotation.internalAutowiredAnnotationProcessororg.springframework.context.annotation.internalCommonAnnotationProcessororg.springframework.context.event.internalEventListenerProcessororg.springframework.context.event.internalEventListenerFactorymyConfigcom.fx.pojo.Applecom.fx.pojo.Orangecom.fx.pojo.Banana
从这里我们可以看到通过@import注入的类如果通过类名获取需要通过类的全路径名称获取,否者会报错
2、注入实现了importSelector接口的类
有的时候我们有这样的业务需求:需要按照操作系统的类型来导入不同的类,例如在Windows系统下导入Apple类,在Linux系统下导入Banana类
importSelector字面上的意思就是导入选择器,根据条件导入不同的类。
怎么让不同的类可以通过同一个类名或者类的字节码文件获得呢?我们可以定义一个接口让我们想要分类的类实现这个接口,再利用多态进行转换
栗子:
定义一个接口:
public interface Fruit {}
让我们想要选择的类实现这个接口:
public class Apple implements Fruit{}
public class Banana implements Fruit{}
写一个类实现importSelector接口,并实现里面选择的方法selectimports
public class FruitimportSelector implements importSelector { @Override public String[] selectimports(Annotationmetadata importingClassmetadata) { //importingClassmetadata里面保存调用此选择器的类 System.out.println("在PearimportSelector中"+importingClassmetadata.toString()); //拿到操作系统名称 String osName = System.getProperty("os.name"); if(osName.contains("Windows")){ //返回Apple类 return new String[]{Apple.class.getName()}; }else if(osName.contains("Linux")){ //返回Banana类 return new String[]{Banana.class.getName()}; } return null; }}
这时候我们要解析类中要导入的就是选择器类了
@Configuration@import({FruitimportSelector.class})public class myConfig {}
测试一下,输出的是:
com.fx.pojo.Apple
当然我们可以通过注入这两个类的接口类获得具体的类
@Testpublic void test(){ ApplicationContext context = context = new AnnotationConfigApplicationContext(myConfig.class); String name = context.getBean(Fruit.class).getClass().getName(); System.out.println(name);}
输出:
com.fx.pojo.Apple
3、注入实现了importBeanDefinitionRegistrar接口的类
可以看到通过@import注入的类如果我们要通过类的全路径名才能从IOC容器拿到类的实例,这样比较麻烦,所以importBeanDefinitionRegistrar接口正是用来给类其别名,简化IOC容器中类的名称的
栗子:
先定义一个类实现下importBeanDefinitionRegistrar接口
public class FruitNameimportBeanDefinitionRegistrar implements importBeanDefinitionRegistrar { @Override public void registerBeanDefinitions(Annotationmetadata importingClassmetadata, BeanDefinitionRegistry beanDefinitionRegistry) { //判断下Apple类是否在容器类了,beanDefinitionRegistry保存了IOC容器中所有Bean的定义项 boolean flag = beanDefinitionRegistry.containsBeanDefinition("com.fx.pojo.Apple"); if(flag){ //先拿到bean的定义项,在Spring容器中每个Bean都被包装成了一个RootBeanDefinition RootBeanDefinition r = new RootBeanDefinition(Apple.class); //将类的访问名字从容器中修改一下 beanDefinitionRegistry.registerBeanDefinition("Apple",r); } }}
注入一下这个类:
@Configuration@import({Apple.class,FruitNameimportBeanDefinitionRegistrar.class})public class myConfig {}
测试:
@Testpublic void test(){ ApplicationContext context = context = new AnnotationConfigApplicationContext(myConfig.class); String[] beanDefinitionNames = context.getBeanDefinitionNames(); Arrays.asList(beanDefinitionNames).forEach(System.out::println);}
打印结果:
Apple
通过这种方式就可以修改通过获得Bean实例的名字了
通过这种方式就可以修改通过获得Bean实例的名字了