談談Spring boot 啟動層面的開發
Spring boot的啟動可以主要分為2個階段。1 是調用AbstractApplicationContext的refresh方法之前和調用AbstractApplicationContext的refresh。 我們知道AbstractApplicationContext的refresh的方法是一個模板方法。幾乎所有類型的ApplicationContext的初始化都是圍繞這個refresh方法來進行。
1. refresh方法之前
1.1 ApplicationContextInitializer
這個類Spring boot***進行調用的類,其主要就是初始化一些BeanFactoryPostProcessor(后面會說明),或者一些在Application 初始化的時候就需要做的事情。而這些類通過通過掃描calsspath路徑下的:META-INF/spring.factories 文件中的org.springframework.context.ApplicationContextInitializer字來加載類的全路徑名,通過反射獲取對象,然后調用initialize方法。
1.2 ApplicationListener
這個和ApplicationContextInitializer加載的方式類似,也是從META-INF/spring.factories文件中的配置,主要作用就是在Spring boot的初始化不同階段會處罰不同的事件(ApplicationEvent及其子類),而這些監聽器就會根據自己在不同事件觸發的情況下完成自己的處理邏輯。例如,ConfigFileApplicationListener負責加載配置文件。
當然也可以通過其他方式加入beanFactory中,詳情可以參照BeanFactoryPostProcessor加入到beanFactory的方法
1.ApplicationListener的有些事件是在ApplicationContextInitializer之前觸發的
2.建議不要直接在ApplicationContextInitializer加入自己的BeanFactoryPostProcessor方法,xxxAware是不會幫你注入的。
2. refresh方法
這個就是spring的模板方法,主要有3個比較重要的接口
2.1 BeanFactoryPostProcessor
public interface BeanFactoryPostProcessor {
void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;
}
這個方法就是給可以對beanFactory進行一些自定義的操作,例如加入一些bean等。當然前提就是我們定義的bean是在beanFactory中。有許多辦法可以做到
- 在ApplicationContextInitializer中定義加入,例如
public class MyContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext> {
public void initialize(ConfigurableApplicationContext applicationContext) {
applicationContext.addBeanFactoryPostProcessor(new MyBeanFactoryPostProcessor());
}
}
- 通過其他BeanFactoryPostProcessor加入,可以通過注解@Import或者加載XML)
例如通過@Import(value={AnnotationBeanDefinitionRegistrar.class})
@Configuration
public class AnnotationBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
private String BEAN_NAME = "annotationBeanPostProcessor";
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
List<String> basePackages = getPackagesToScan(importingClassMetadata);
if (!registry.containsBeanDefinition(BEAN_NAME)) {
addPostProcessor(registry, basePackages);
}
}
private void addPostProcessor(BeanDefinitionRegistry registry, List<String> basePackages) {
GenericBeanDefinition beanDefinition = new GenericBeanDefinition();
beanDefinition.setBeanClass(AnnotationBeanPostProcessor.class);
beanDefinition.getConstructorArgumentValues()
.addGenericArgumentValue(basePackages);
beanDefinition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
registry.registerBeanDefinition(BEAN_NAME, beanDefinition);
}
private List<String> getPackagesToScan(AnnotationMetadata metadata) {
AnnotationAttributes attributes = AnnotationAttributes.fromMap(
metadata.getAnnotationAttributes(EnableDubbo.class.getName()));
String[] basePackages = attributes.getStringArray("basePackages");
return Arrays.asList(basePackages);
}
}
這樣,就可以加入我們自定義的BeanFactoryPostProcessor,就可以在Bean注冊的層面上進行開發
2.2 BeanPostProcessor
public interface BeanPostProcessor {
Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException;
Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException;
}
這個接口主要就是針對在bean實例化前后做一些定制開發。一般只針對某個接口或者某個注解進行批量操作
3. 綜述
1. 如果你需要在SpringApplication初始化的時候就做一些事情,使用ApplicationContextInitializer
2. 如果你需要SpringApplication的某個特定階段做一些事情,使用ApplicationListener(推薦)
3. 如果你需要在beanFactory層面上開發,使用BeanFactoryPostProcessor(推薦)
4. 如果你需要在對某個bean的實例化層面上開發,使用BeanPostProcessor(一般業務上的開發使用InitializingBean或者init-method能夠滿足)
轉載請注明出處:https://my.oschina.net/u/3039671/blog/852211
例子,Spring boot風格使用dubbo
碼云:https://git.oschina.net/null_584_3382/spring-dubbo-parent