【Spring】Spring後置處理器

活在夢裡丶發表於2020-12-23

Spring經過XML解析獲取到了所有需要例項化Bean的資訊並讀取到BeanDefinition後,就儲存在了BeanDefinitionMap容器中。在例項化bean的操作就是依據這些BeanDefinition定義來做,而在例項化之前,Spring允許我們通過自定義擴充套件來改變bean的定義。定義一旦變了,後面的例項也就變了。而beanFactory後置處理器,既BeanFactoryPostProcessor就是用來改變bean定義的。

invokeBeanFactoryPostProcessors(beanFactory)

// 建立BeanFactory,並且把定義的Bean的資訊讀取到了BeanDefinition中
invokeBeanFactoryPostProcessors(beanFactory);

// BeanDefinitionRegistryPostProcessor 和 BeanFactoryPostProcessor介面的呼叫
ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();

在載入完BeanDefinition後緊接著就會執行obtainFreshBeanFactory,幫我們呼叫實現了BeanFactoryPostProcessor和BeanDefinitionRegistryPostProcessor介面的方法。

BeanDefinitionRegistryPostProcessor繼承了BeanFactoryPostProcessor介面,所以它們一共有兩個核心的方法:

@Component
public class WintigBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor {

    /**
     * 在例項化之前,可以對BeanDefinition進行修改和註冊
     * @param registry the bean definition registry used by the application context
     * @throws BeansException
     */
    @Override
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {

        // 獲取所有beanDefinitionNames
        String[] beanDefinitionNames = registry.getBeanDefinitionNames();

        for (String beanDefinitionName : beanDefinitionNames) {
            // 根據name獲取beanDefinition
            BeanDefinition beanDefinition = registry.getBeanDefinition(beanDefinitionName);
        }

        // 新增beanDefinition
        GenericBeanDefinition genericBeanDefinition = new GenericBeanDefinition();
        genericBeanDefinition.setBeanClass(GenericBeanDefinitionBean.class);
        // 屬性設定
        MutablePropertyValues propertyValues = genericBeanDefinition.getPropertyValues();
        propertyValues.add("name", "wintig");
        registry.registerBeanDefinition("genericBeanDefinitionBean", genericBeanDefinition);

        // 刪除指定beanDefinition
        registry.removeBeanDefinition("genericBeanDefinitionBean");

        // 自定義掃描器
        // 就完成了"com.wintig"路徑下,有WintigService註解類的掃描
        ClassPathBeanDefinitionScanner scanner = new ClassPathBeanDefinitionScanner(registry);
        scanner.addIncludeFilter(new AnnotationTypeFilter(WintigService.class));
        scanner.scan("com.wintig");
    }

    /**
     * 拿到beanFactory容器,你可以根據業務進行擴充套件
     * @param beanFactory the bean factory used by the application context
     * @throws BeansException
     */
    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        DefaultListableBeanFactory listableBeanFactory = (DefaultListableBeanFactory)beanFactory;
        // 允許使用相同名稱重新註冊不同的bean實現
        listableBeanFactory.setAllowBeanDefinitionOverriding(true);
        // 允許迴圈依賴
        listableBeanFactory.setAllowCircularReferences(true);
    }

    public class GenericBeanDefinitionBean {
        
        private String name;

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }
    }
}

 

相關文章