Spring 与组件注入相关的注解之 @Import
前言:
前面已经介绍了十好几个与组件注入相关的注解了,这里再介绍一个,但也是在 springboot 源码中经常出现的一个,因此还是比较重要的。
注解简介:
@Import 也是用于向 IOC 容器中注入相应组件的。通过查看 @Import 的源码:
该注解需要标注在类上,且仅有一个 value 属性,value 属性的注释信息已经做了很好的解释,即 value 属性
- 要么是普通的组件(标识了 @Component 注解等),
- 要么是 ImportSelector 接口的实现类,
- 要么是 ImportBeanDefinitionRegistrar 接口的实现类。
实验:
第一种方式:
- 自定义一个类,例如 Teacher 等等;
- 在配置类上标注上 @Import 注解,就可以实现 Teacher 组件的注入:
@Import(value={Teacher.class})
public class SpringConfig2 {
}
这种方式默认注入的组件的id为全类名,例如 com.uestc.auto.xiaoxie.bean.Teacher。
第二种方式:
- 自定义一个实现了 ImportSelector 接口(它可以帮助我们根据导入特定的组件)的类,如 MyImportSelector;
- 重写接口的 selectImports() 方法;
// 自定义逻辑导入特定的组件
public class MyImportSelector implements ImportSelector{
/**
* Select and return the names of which class(es) should be imported based on
* the {@link AnnotationMetadata} of the importing @{@link Configuration} class.
*
* AnnotationMetadata: 标注 @Import 注解的类的所有注解信息(稍后通过 debug 源码也可以看到)
*/
@Override
public String[] selectImports(AnnotationMetadata importingClassMetadata) {
Set<String> annotationTypes = importingClassMetadata.getAnnotationTypes();
System.out.println("=============");
System.out.println(annotationTypes); // 打印结果:[org.springframework.context.annotation.Configuration, org.springframework.context.annotation.ComponentScans, org.springframework.context.annotation.Import]
System.out.println("=============");
// 该方法的返回值为需要导入的组件的全类名。
return new String[]{"com.uestc.auto.xiaoxie.bean.Professor", "com.uestc.auto.xiaoxie.bean.Student"}; // 在 return 前面打上断点,进行调试
}
}
需要额外说明一点:selectImports() 方法中如果不注入任何组件的话,最好返回一个空数组;否则会抛出空指针异常。原因如下图所示,debug 源码的时候,需要现在 ApplicationContext 的创建(我是写在单元测试的 beforeMethod 方法中了)以及 selectImports() 方法的 return 语句打上两个断点:
3. 在配置类上的 @Import 注解中添加上这个类:@Import(value={Teacher.class, MyImportSelector.class}),就可以再将 Professor 和 Student 这两个类注入到容器中。
第三种方式:
- 自定义一个实现了 ImportBeanDefinitionRegistrar 接口的实现类,如 MyImportBeanDefinitionRegister;
- 重写接口的 registerBeanDefinitions() 方法;
public class MyImportBeanDefinitionRegister implements ImportBeanDefinitionRegistrar{
/*
* AnnotationMetadata: 标注 @Import 注解的类的所有注解信息
* BeanDefinitionRegistry: Bean定义的注册类
*/
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
// Register a new bean definition with this registry.
// Must support RootBeanDefinition and ChildBeanDefinition.
// registry.registerBeanDefinition(String beanName, BeanDefinition beanDefinition);
registry.registerBeanDefinition("boy", new RootBeanDefinition(Boy.class)); // 依然可以仿照前面的思路,在这一行的前面打断点,观察是何时执行的。
}
}
- 在配置类上的 @Import 注解中添加上这个类:@Import(value={Teacher.class, MyImportSelector.class, MyImportBeanDefinitionRegister.class}),就可以再将 Boy 这两个组件注入到容器中。
小结:
到目前为止,已经总结了很多与组件注入相关的注解。建议多写几个 demo 分开进行测试,免得影响测试结果,进一步影响对注解的理解。 另外,用好 debug,这个真的太重要了!!!