Spring进阶篇(5)- BeanPostProcessor(Bean的后置处理器)

JAVA && Spring && SpringBoot2.x — 学习目录

BeanPostProcessor在字面上理解为:Bean的后置处理器,允许Spring框架在创建Bean时对其进行定制化的修改。

经典的应用:初始化Bean时创建代理对象,初始化Bean时读取@Autowired等注解。

注意点:BeanPostProcessor本身也是一个Bean,而它的初始化时机一般要早于普通的Bean。如高优先级BeanPostProcessor中依赖一些普通Bean时,可能会造成普通的Bean实例化时机早于低优先级BeanPostProcessor,造成低优先的BeanPostProcessor不能生效。

1. 自定义实现BeanPostProcessor接口

@Slf4j
@Component
public class MyBeanPostProcessor implements BeanPostProcessor,Ordered{
    //测试用,在自定义BeanPostProcessor中引入普通Bean对象
    @Autowired
    private AccountImpl account;

    @Override
    public int getOrder() {
        return Ordered.LOWEST_PRECEDENCE;
    }

   //实例化、依赖注入完毕,在调用显示的初始化之前完成一些定制的初始化任务 
   @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        if (beanName.contains("aServiceImpl")) {
            log.info("【bean初始化之前,bean is {},beanName is {}】", bean.toString(), beanName);
        }
        return bean;
    }
    //实例化、依赖注入、初始化完毕时执行  
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        if (beanName.contains("aServiceImpl")) {
            log.info("【bean初始化之后,bean is {},beanName is {}】", bean.toString(), beanName);
        }
        return bean;
    }
}

注:接口的两个方法都不能返回null,因为后置处理器从Spring IOC容器中获取Bean实例对象没有再次放回到IOC容器中,后续操作获取Bean对象可能会空指针异常!

启动项目后,在调用Bean对象时,发现本应被代理的Bean对象没有被代理。

service代码失去代理.png

实际上,在自定义的BeanPostProcessor中依赖注入Bean对象,那么Bean对象没有经过代理的BeanPostProcessor,造成问题。那么BeanPostProcessor的加载优先级如何呢?

2. BeanPostProcessor加载的优先级

一个Bean可能包含多个后置处理器,如何控制不同的BeanPostProcessor的加载顺序呢?

org.springframework.context.support.PostProcessorRegistrationDelegate#registerBeanPostProcessors(ConfigurableListableBeanFactory, AbstractApplicationContext)中,对BeanPostProcessor的加载分成了4个部分:

public static void registerBeanPostProcessors(
            ConfigurableListableBeanFactory beanFactory, AbstractApplicationContext applicationContext) {

        // 从容器中获取所有的BeanPostProcessor类型的BeanName数组
        String[] postProcessorNames = beanFactory.getBeanNamesForType(BeanPostProcessor.class, true, false);

        // Register BeanPostProcessorChecker that logs an info message when
        // a bean is created during BeanPostProcessor instantiation, i.e. when
        // a bean is not eligible for getting processed by all BeanPostProcessors.
        int beanProcessorTargetCount = beanFactory.getBeanPostProcessorCount() + 1 + postProcessorNames.length;
        beanFactory.addBeanPostProcessor(new BeanPostProcessorChecker(beanFactory, beanProcessorTargetCount));

        // 分离实现了PriorityOrdered,Ordered 接口的BeanPostProcessors 和其他类型的BeanPostProcessors
        List<BeanPostProcessor> priorityOrderedPostProcessors = new ArrayList<>();
        List<BeanPostProcessor> internalPostProcessors = new ArrayList<>();
        List<String> orderedPostProcessorNames = new ArrayList<>();
        List<String> nonOrderedPostProcessorNames = new ArrayList<>();
        for (String ppName : postProcessorNames) {
            if (beanFactory.isTypeMatch(ppName, PriorityOrdered.class)) {
                BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class);
                priorityOrderedPostProcessors.add(pp);
                if (pp instanceof MergedBeanDefinitionPostProcessor) {
                    internalPostProcessors.add(pp);
                }
            }
            // 如果实现了 Ordered 接口,则将全类名添加到orderedPostProcessorNames的List中
            else if (beanFactory.isTypeMatch(ppName, Ordered.class)) {
                orderedPostProcessorNames.add(ppName);
            }
            // 否则添加全类名到 nonOrderedPostProcessorNames 的List 中
            else {
                nonOrderedPostProcessorNames.add(ppName);
            }
        }

        //首先,注册实现了PriorityOrdered接口的接口
        sortPostProcessors(priorityOrderedPostProcessors, beanFactory);
        registerBeanPostProcessors(beanFactory, priorityOrderedPostProcessors);

        // 然后,注册实现了Ordered 接口的后置处理器
        List<BeanPostProcessor> orderedPostProcessors = new ArrayList<>();
        for (String ppName : orderedPostProcessorNames) {
            BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class);
            orderedPostProcessors.add(pp);
            if (pp instanceof MergedBeanDefinitionPostProcessor) {
                internalPostProcessors.add(pp);
            }
        }
        sortPostProcessors(orderedPostProcessors, beanFactory);
        registerBeanPostProcessors(beanFactory, orderedPostProcessors);

        // 注册所有常规的 BeanPostProcessors
        List<BeanPostProcessor> nonOrderedPostProcessors = new ArrayList<>();
        for (String ppName : nonOrderedPostProcessorNames) {
            BeanPostProcessor pp = beanFactory.getBean(ppName, BeanPostProcessor.class);
            nonOrderedPostProcessors.add(pp);
            if (pp instanceof MergedBeanDefinitionPostProcessor) {
                internalPostProcessors.add(pp);
            }
        }
        registerBeanPostProcessors(beanFactory, nonOrderedPostProcessors);

        // 最后, 再次注册所有的内部BeanPostProcessors
        sortPostProcessors(internalPostProcessors, beanFactory);
        registerBeanPostProcessors(beanFactory, internalPostProcessors);

        // Re-register post-processor for detecting inner beans as ApplicationListeners,
        // moving it to the end of the processor chain (for picking up proxies etc).
        beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(applicationContext));
    }
  1. 加载实现PriorityOrdered接口的BeanPostProcessor类;
  2. 加载实现Ordered接口的BeanPostProcessor类;
  3. 加载BeanPostProcessor类;
  4. 加载实现MergedBeanDefinitionPostProcessor接口的BeanPostProcessor类;

高优先级的BeanPostProcessor类可以影响低优先级的BeanPostProcessor类,但是同优先级的BeanPostProcessor不能相互作用。

即若是Ordered级别的BeanPostProcessor实现引用了Bean对象,那么该Bean对象将失去所有Order级别后置处理器的处理。

因此在实现BeanPostProcessor接口时,应注意加载顺序的影响。

3. Spring的初始执行顺序

BeanFactoryPostProcessor的postProcessBeanFactory方法
构造方法!
BeanPostProcessor的postProcessBeforeInitialization方法
声明@PostConstruct注解的方法
InitializingBean接口的afterPropertiesSet()方法
声明的init方法!
BeanPostProcessor的postProcessAfterInitialization方法
容器启动—CommandLineRunner接口方法!

BeanPostProcessor的两个处理方法,是在init方法前后进行处理的。

推荐阅读

BeanPostProcessor加载次序及其对Bean造成的影响分析

【小家Spring】注意BeanPostProcessor启动时对依赖Bean的“误伤”陷阱

实战—BeanPostProcessor自定义bean加载后属性注入

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352