简单分析BeanPostProcessor

Posted kingsleylam

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了简单分析BeanPostProcessor相关的知识,希望对你有一定的参考价值。

1. 什么是BeanPostProcessor
BeanPostProcessor是一个接口,有两个方法,分别是:Object postProcessBeforeInitialization(Object var1, String var2) throws BeansException 和 Object postProcessAfterInitialization(Object var1, String var2) throws BeansException;
Spring Bean的生命周期中,在为Bean实例化,装配好属性后,会调用上下文中所有的BeanPostProcessor对象的两个方法为其初始化;

2. 一个小例子
分别创建三个类,分别是接口类、接口类的实现类,和BeanPostProcessor的实现类。

1 package com.khlin.my.test;
2 
3 public interface WelcomeService 
4 
5     void welcome();
6 
 1 package com.khlin.my.test;
 2 
 3 import org.springframework.beans.factory.InitializingBean;
 4 
 5 public class WelcomeServiceImpl implements WelcomeService, InitializingBean 
 6 
 7     public void init() 
 8         System.out.println("init.");
 9     
10 
11     public void welcome() 
12         System.out.println("Welcome to Spring.");
13     
14 
15     public void afterPropertiesSet() throws Exception 
16         System.out.println("afterPropertiesSet.");
17     
18 
 1 package com.khlin.my.test;
 2 
 3 import org.springframework.beans.BeansException;
 4 import org.springframework.beans.factory.config.BeanPostProcessor;
 5 
 6 public class LoginProcessor implements BeanPostProcessor 
 7 
 8     public Object postProcessBeforeInitialization(Object o, String s) throws BeansException 
 9         System.out.println("login successfully.");
10         return o;
11     
12 
13     public Object postProcessAfterInitialization(Object o, String s) throws BeansException 
14         System.out.println("logout successfully.");
15         return o;
16     
17 

 

在applicationContext.xml里实例化对应的Bean.

1 <bean id="welcomeService" class="com.khlin.my.test.WelcomeServiceImpl" init-method="init"/>
2 <bean id="loginProcessor" class="com.khlin.my.test.LoginProcessor"/>

 

再写一个启动类

 1 package com.khlin.my.test;
 2 
 3 import org.springframework.context.ApplicationContext;
 4 import org.springframework.context.support.ClassPathXmlApplicationContext;
 5 
 6 public class IOCTest 
 7 
 8     public static void main(String[] args) 
 9         ApplicationContext context = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
10         System.out.println("context 启动成功");
11         WelcomeService messageService = context.getBean(WelcomeService.class);
12         messageService.welcome();
13     
14 

 启动main方法,输出如下:

技术图片

调用顺序分别为:

BeanPostProcessor的postProcessBeforeInitialization

InitializingBean的afterPropertiesSet

init方法

BeanPostProcessor的postProcessAfterInitialization

3. 实现原理

我们来看一下Spring启动一个上下文的时候,都做了啥。这里不作详细的源码解读。

技术图片

 

可以看到上下文ApplicationContext持有一个BeanFactory。在第一个红框,将所有的BeanPostProcessor注册到BeanFactory。

调试代码,可以看到注册后保存在BeanFactory的beanPostProcessors集合里。

技术图片

再来看第二个红框,finishBeanFactoryInitialization() 这个方法会对Bean进行初始化。

在AbstractAutowireCapableBeanFactory这个类的protected Object initializeBean(final String beanName, final Object bean, RootBeanDefinition mbd)方法里,可以看到invokeInitMethods方法被夹在BeanPostProcessor两个方法的中间。

技术图片

在applyBeanPostProcessorsBeforeInitialization和applyBeanPostProcessorsAfterInitialization方法中,会遍历所有注册的BeanBeanProcessor并调用方法。

每个BeanPostProcessor可以返回处理后的对象,如果返回null,会导致遍历中断,可能有些BeanPostProcessor无法处理,这点要注意。

技术图片

 

InitMethod有两种方式,一种是在配置文件中加上init-method属性并指定对应的方法,另一种是实现InitializingBean接口的afterPropertiesSet()方法。

可以看到是优先调用afterPropertiesSet()方法,再调用init-method指定的方法,这与我们的输出顺序一致。

技术图片

 

以上是关于简单分析BeanPostProcessor的主要内容,如果未能解决你的问题,请参考以下文章

简单算法分析2 ------适合新手学习算法分析

Spring源码分析-2 Spring的简单使用和原理分析

Spring源码分析-2 Spring的简单使用和原理分析

用python怎么做方差分析的简单主效应分析

Leetcode动态规划简单题

FFmpeg源码简单分析:结构体成员管理系统-AVOption