在application context执行完标准初始化之后,所有bean definition都已加载,但还没有实例化任何bean。BeanFactoryPostProcessor允许插手修改context内部bean工厂,允许覆盖或添加属性。
public class Test {
public static void main(String[] args) {
AnnotationConfigApplicationContext context =
new AnnotationConfigApplicationContext();
context.register(AppConfig.class);
context.refresh();
UserDao dao = (UserDao) context.getBean("userDao");
UserDao dao1 = (UserDao) context.getBean("userDao");
System.out.println("dao:"+dao.hashCode()+"->dao1:"+dao1.hashCode());
dao.query();
}
}
@Repository
public class UserDao {
public void query(){
System.out.println("UserDao query()");
}
}
@Component
public class TestBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
BeanDefinition beanDefinition =beanFactory.getBeanDefinition("userDao");
beanDefinition.setScope("prototype");
}
}
执行结果如下:
改变了UserDao的scope属性。
需要注意的是,BeanFactoryPostProcessor接口有一个重要的子接口BeanDefinitionRegistryPostProcessor
spring在执行context.fresh()时对这两个接口的用处给区分开来,BeanDefinitionRegistryPostProcessor的postProcessBeanDefinitionRegistry()方法对BeanFactoryPostProcessor进行了扩展。