Spring5源码(10)-BeanPostProcessor和BeanFactoryPostProcessor的区别

简介: Spring5源码(10)-BeanPostProcessor和BeanFactoryPostProcessor的区别


  • BeanPostProcessor接口:后置bean处理器,允许自定义修改新的bean实例,应用程序上下文可以在其bean定义中自动检测BeanPostProcessor类型的bean,并将它们应用于随后创建的任何bean。(例如:配置文件中注册了一个自定义BeanPostProcessor类型的bean,一个User类型的bean,应用程序上下文会在创建User实例之后对User应用BeanPostProcessor)。
  • BeanFactoryPostProcessor接口:后置工厂处理器,允许自定义修改应用程序上下文的bean定义,调整bean属性值。应用程序上下文可以在其bean定义中自动检测BeanFactoryPostProcessor,并在创建任何非BeanFactoryPostProcessor类型bean之前应用它们(例如:配置文件中注册了一个自定义BeanFactoryPostProcessor类型的bean,一个User类型的bean,应用程序上下文会在创建User实例之前对User应用BeanFactoryPostProcessor)。
对比 BeanFactoryPostProcessor BeanPostProcessor
回调时间 Bean实例化完成之前 Bean实例化完成之后
是否可修改bean定义信息
是否可修改bean实例信息
是否支持排序接口
方法级别 ApplicationContext级别 ApplicationContext级别
实现方式 注册自定义BeanFactoryPostProcessor,实现BeanFactoryAware接口
注册自定义BeanPostProcessor
1. BeanPostProcessor
  • bean

package com.lyc.cn.v2.day02.processor.bean;
import com.lyc.cn.v2.day01.Dog;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.core.Ordered;
/**
 * 继承BeanPostProcessor接口并重写其方法
 * @author: LiYanChao
 * @create: 2018-09-06 14:56
 */
public class MyBeanPostProcessorOne implements BeanPostProcessor, Ordered {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("BeanPostProcessor第" + getOrder() + "次被调动\n");
        if (bean instanceof Dog) {
            ((Dog) bean).setName("强强");
        }
        return bean;
    }
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        if (bean instanceof Dog) {
            ((Dog) bean).setAge(5);
        }
        return bean;
    }
    @Override
    public int getOrder() {
        return 1;
    }
}

package com.lyc.cn.v2.day02.processor.bean;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.core.Ordered;
/**
 * 继承BeanPostProcessor接口并重写其方法
 * @author: LiYanChao
 * @create: 2018-09-06 14:56
 */
public class MyBeanPostProcessorTwo implements BeanPostProcessor, Ordered {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("BeanPostProcessor第" + getOrder() + "次被调动");
        return bean;
    }
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }
    @Override
    public int getOrder() {
        return 2;
    }
}
2. BeanFactoryPostProcessor

package com.lyc.cn.v2.day02.processor.factory;
import org.springframework.beans.BeansException;
import org.springframework.beans.MutablePropertyValues;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.core.Ordered;
public class MyBeanFactoryPostProcessorOne implements BeanFactoryPostProcessor, Ordered {
    /**
     * 修改dog的name属性值
     * 修改dog的作用域
     */
    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        System.out.println("BeanFactoryPostProcessor第" + getOrder() + "次被调动");
        BeanDefinition bd = beanFactory.getBeanDefinition("dog");
        if (null != bd) {
            System.out.println("dog属性值:" + bd.getPropertyValues().toString());
            MutablePropertyValues pv = bd.getPropertyValues();
            if (pv.contains("name")) {
                System.out.println("修改dog的name属性值为强强");
                pv.addPropertyValue("name", "强强");
            }
            System.out.println("修改dog的作用域为prototype\n");
            bd.setScope(BeanDefinition.SCOPE_PROTOTYPE);
        }
    }
    @Override
    public int getOrder() {
        return 0;
    }
}

package com.lyc.cn.v2.day02.processor.factory;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.core.Ordered;
public class MyBeanFactoryPostProcessorTwo implements BeanFactoryPostProcessor, Ordered {
    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        System.out.println("BeanFactoryPostProcessor第" + getOrder() + "次被调动");
    }
    @Override
    public int getOrder() {
        return 1;
    }
}
  • xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd" profile="dev">
    <bean id="dog" class="com.lyc.cn.v2.day01.Dog">
        <!-- 指定构造器参数 index对应构造器中参数的位置 -->
        <!-- 也可以通过指定参数类型,指定参数名称来注入属性-->
        <constructor-arg index="0" value="小明"/>
        <constructor-arg index="1" value="3"/>
    </bean>
    <!-- =====beanPostProcessor配置begin =====-->
    <bean id="myBeanPostProcessorOne" class="com.lyc.cn.v2.day02.processor.bean.MyBeanPostProcessorOne"/>
    <bean id="myBeanPostProcessorTwo" class="com.lyc.cn.v2.day02.processor.bean.MyBeanPostProcessorTwo"/>
    <!-- =====beanPostProcessor配置end =====-->
    <!-- =====beanFactoryPostProcessor配置begin =====-->
    <!--注册第一个后置工厂处理器-->
    <bean id="myBeanFactoryPostProcessorOne" class="com.lyc.cn.v2.day02.processor.factory.MyBeanFactoryPostProcessorOne"/>
    <!--注册第二个后置工厂处理器-->
    <bean id="myBeanFactoryPostProcessorTwo" class="com.lyc.cn.v2.day02.processor.factory.MyBeanFactoryPostProcessorTwo"/>
    <!-- =====beanFactoryPostProcessor配置end =====-->
</beans>
  • 测试

package com.lyc.cn.v2.day02;
import com.lyc.cn.v2.day01.Dog;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.core.io.ClassPathResource;
/**
 * @author: LiYanChao
 * @create: 2018-09-07 23:40
 */
public class MyTest {
    private XmlBeanFactory xmlBeanFactory;
    @Before
    public void initXmlBeanFactory() {
        System.setProperty("spring.profiles.active", "dev");
        System.out.println("\n========测试方法开始=======\n");
        xmlBeanFactory = new XmlBeanFactory(new ClassPathResource("v2/day01.xml"));
    }
    @After
    public void after() {
        System.out.println("\n========测试方法结束=======\n");
    }
    @Test
    public void test1() {
        // 测试BeanPostProcessor
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("v2/day02.xml");
        Dog dog = applicationContext.getBean("dog", Dog.class);
        dog.sayHello();
    }
    @Test
    public void test2() {
        // 测试BeanFactoryPostProcessor
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("v2/day02.xml");
        Dog dog = applicationContext.getBean("dog", Dog.class);
        dog.sayHello();
    }
}
  • 结果
    为了不影响测试结果,需要将BeanPostProcessor和BeanFactoryPostProcessor分开测试,测试test1时将test2的xml配置注释掉,反之亦然。

test1-->

========测试方法开始=======
BeanPostProcessor第1次被调动
BeanPostProcessor第2次被调动
大家好, 我叫强强, 我今年5岁了
========测试方法结束=======

test2-->

========测试方法开始=======
BeanFactoryPostProcessor第0次被调动
dog属性值:PropertyValues: length=0
修改dog的作用域为prototype
BeanFactoryPostProcessor第1次被调动
大家好, 我叫小明, 我今年3岁了
========测试方法结束=======





目录
相关文章
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
110 2
|
2月前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
110 5
|
13天前
|
监控 JavaScript 数据可视化
建筑施工一体化信息管理平台源码,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
智慧工地云平台是专为建筑施工领域打造的一体化信息管理平台,利用大数据、云计算、物联网等技术,实现施工区域各系统数据汇总与可视化管理。平台涵盖人员、设备、物料、环境等关键因素的实时监控与数据分析,提供远程指挥、决策支持等功能,提升工作效率,促进产业信息化发展。系统由PC端、APP移动端及项目、监管、数据屏三大平台组成,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
|
1月前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
64 2
|
1月前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
2月前
|
前端开发 Java 开发者
Spring生态学习路径与源码深度探讨
【11月更文挑战第13天】Spring框架作为Java企业级开发中的核心框架,其丰富的生态系统和强大的功能吸引了无数开发者的关注。学习Spring生态不仅仅是掌握Spring Framework本身,更需要深入理解其周边组件和工具,以及源码的底层实现逻辑。本文将从Spring生态的学习路径入手,详细探讨如何系统地学习Spring,并深入解析各个重点的底层实现逻辑。
79 9
|
3月前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
215 5
|
3月前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
3月前
|
XML Java 数据格式
手动开发-简单的Spring基于注解配置的程序--源码解析
手动开发-简单的Spring基于注解配置的程序--源码解析
59 0
|
3月前
|
XML Java 数据格式
手动开发-简单的Spring基于XML配置的程序--源码解析
手动开发-简单的Spring基于XML配置的程序--源码解析
96 0