SpringBoot -【BeanPostProcessor】基础使用及应用场景

news2025/5/21 3:40:35

BeanPostProcessor应用与优化

1. 引言

在现代软件开发中,企业开发面临着越来越复杂的系统架构和业务需求。随着项目规模的扩大和技术栈的增多,需要更高效的工具来应对这些挑战,并确保代码的可维护性和扩展性。

在这样的背景下,BeanPostProcessor作为一个重要的工具,为程序员提供了一种灵活且强大的方式来管理和组织代码。BeanPostProcessor不仅简化了依赖管理和配置管理的复杂性,还提供了一种结构化的方式来实现面向切面编程(AOP),实现代码的模块化和可重用性。

BeanProcessor主要应用在以下几个方面:

  1. 依赖注入的简化: 通过BeanPostProcessor,我们可以轻松实现依赖注入,减少了手动管理对象之间的依赖关系的复杂性,提高了代码的可测试性和可维护性。
  2. 面向切面编程(AOP)的支持: BeanPostProcessor提供了AOP的支持,使得我们可以更加灵活地实现横切关注点,如日志记录、事务管理等,将这些与核心业务逻辑分离开来,提高了代码的模块化程度。
  3. 配置管理的优化: 通过BeanPostProcessor,我们可以将配置信息与代码分离,实现了配置的集中管理和动态加载,降低了系统的耦合度,使得系统更易于维护和扩展。

2. BeanPostProcessor概述

2.1 概述

Spring框架中,BeanPostProcessor接口是一个重要的接口,它定义了两个关键的方法:postProcessBeforeInitializationpostProcessAfterInitialization。这两个方法允许开发人员在Bean对象的初始化过程中插入自定义逻辑,从而对Bean进行定制化的处理。

1. postProcessBeforeInitialization方法

postProcessBeforeInitialization方法在Bean对象初始化之前被调用。开发人员可以利用这个方法在Bean初始化之前执行一些自定义的逻辑。下面是这个方法的一些关键点:

  • 功能:允许开发人员在Bean初始化之前对Bean对象进行预处理。
  • 用途:通常用于执行一些初始化前的逻辑,例如:验证、属性设置、数据加载等。
  • 时机:在Bean的属性设置之后,但在初始化方法(如init-method)之前被调用。
  • 返回值:该方法的返回值是一个对象,允许开发人员修改Bean对象的实例,甚至可以返回一个完全不同的实例对象。

2. postProcessAfterInitialization方法

postProcessAfterInitialization方法在Bean对象初始化之后被调用。开发人员可以利用这个方法在Bean初始化之后执行一些自定义的逻辑。以下是这个方法的一些重要信息:

  • 功能:允许开发人员在Bean初始化之后对Bean对象进行后处理。
  • 用途:通常用于执行一些初始化后的逻辑,例如:初始化之后的验证、注册、清理等。
  • 时机:在Bean的初始化方法(如init-method)之后被调用。
  • 返回值:同样是一个对象,允许开发人员修改Bean对象的实例,但返回的对象必须是Bean对象的后代(subclass)或者相同的实例。

这样我们就可以在项目启动前后做一些额外的操作,例如

  • 定制Bean初始化过程:通过实现BeanProcessor接口,并重写其方法,开发人员可以在Bean初始化的不同阶段插入自定义逻辑,实现对Bean的定制化处理。
  • 扩展Spring框架功能:利用这两个方法,开发人员可以在Spring框架的基础上,扩展出更多自定义的功能和特性,以满足具体项目的需求。
  • 灵活处理Bean对象:通过BeanProcessor接口提供的前置和后置初始化方法,开发人员可以对Bean对象进行更加灵活和细粒度的处理,从而满足各种复杂业务逻辑的需求。

2.1 使用BeanPostProcessor

2.1.1 创建BeanPostProcessor实例
/**
 * @author 13723
 * @version 1.0
 * 2024/2/22 17:13
 */
@Component
public class MyCustomerBeanPostProcessor implements BeanPostProcessor {
	private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());

	MyCustomerBeanPostProcessor(){
		logger.error(" 初始化 MyCustomerBeanPostProcessor ! ");
	}

	/**
	 *
	 * 在initializeBean方法里面,先后调用了applyBeanPostProcessorsBeforeInitialization和applyBeanPostProcessorsAfterInitialization方法,
	 * 这两个方法内部,则分别去遍历系统里所有的BeanPostProcessor。然后逐个执行这些BeanPostProcessor对象的postProcessBeforeInitialization和postProcessAfterInitialization方法,去处理对象
	 * 这时 会调用到我们自定义的BeanPostProcessor的postProcessBeforeInitialization和postProcessAfterInitialization方法
	 * @param bean bean实例
	 * @param beanName bean名称
	 * @return bean实例
	 * @throws BeansException 异常
	 * @see  org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#initializeBean(java.lang.String, java.lang.Object, org.springframework.beans.factory.support.RootBeanDefinition)*
	 */
	@Override
	public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
		// 注意 这里会遍历所有的bean
		if (bean instanceof MyCustomerBean){
			logger.error("------ 执行前置 MyCustomerBeanPostProcessor 方法 beanName : {}! ------",beanName);
		}
		return bean;
	}

	/**
	 * 同上,不过时在初始化之后执行
	 * @param bean
	 * @param beanName
	 * @return
	 * @throws BeansException
	 */
	@Override
	public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
		if (bean instanceof MyCustomerBean){
			logger.error("------ 执行后置 MyCustomerBeanPostProcessor 方法 beanName : {}! ------",beanName);
		}
		return bean;
	}
}
2.1.2 创建普通的Bean对象
/**
 * @author 13723
 * @version 1.0
 * 2024/2/22 17:55
 */
@Component
public class MyCustomerBean {
	private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());

	MyCustomerBean(){
		logger.error(" 初始化 MyCustomerBean !");
	}
}

在这里插入图片描述

2.2.注意事项

Ordered接口用来指定多个BeanPostProcessor实现的方法的执行顺序

2.2.1 BeanPostProcessor不能依赖别的Bean
public class MyCustomerBeanPostProcessor implements BeanPostProcessor, Ordered {
	private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());

	/**
	 * 注意1. 当BeanPostProcessor 依赖别的Bean时。别的Bean的初始化方法会在BeanPostProcessor之前执行
	 * 会导致不会执行 BeanPostProcessor 的初始化方法 以及 postProcessBeforeInitialization 方法 以及 postProcessAfterInitialization 方法
	 */
	@Autowired
	private CustomerService customerService;


	MyCustomerBeanPostProcessor(){
		logger.error(" 初始化 MyCustomerBeanPostProcessor ! ");
	}

	@Override
	public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
		// 注意 这里会遍历所有的bean
		if (bean instanceof MyCustomerBean){
			logger.error("------ 执行前置 MyCustomerBeanPostProcessor 方法 beanName : {}! ------",beanName);
		}
		return bean;
	}

	/**
	 * 同上,不过时在初始化之后执行
	 * @param bean
	 * @param beanName
	 * @return
	 * @throws BeansException
	 */
	@Override
	public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
		if (bean instanceof MyCustomerBean){
			logger.error("------ 执行后置 MyCustomerBeanPostProcessor 方法 beanName : {}! ------",beanName);
		}
		return bean;
	}

	@Override
	public int getOrder() {
		return HIGHEST_PRECEDENCE + 1;
	}
}

在这里插入图片描述

3. 应用场景

3.1 企业开发中使用BeanPostProcessor场景

在某些情况下,我们可能需要在应用程序启动时扫描特定的注解,并在Bean初始化时进行处理。例如,我们希望在Spring Boot应用中扫描所有带有特定注解的类,并在它们被初始化时打印日志,和执行他们所有的方法。

3.2 实现思路

  • 创建一个自定义的BeanPostProcessor实现类,扫描带有特定注解的类,并在它们被初始化时执行自定义逻辑。
  • 将该BeanPostProcessor注册到Spring容器中。
  • 编写带有特定注解的测试类,验证BeanPostProcessor的功能是否正常。
3.2.1 创建自定义注解

关于注解的原理可以参考

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface AlarmTarget {
}

3.2.2 创建使用注解的类
@AlarmTarget
@Component
public class CustomerServiceInfo {
	private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());

	public void doSomething(){
		logger.error("------ 通过反射执行了  CustomerServiceInfo 的 doSomething 方法! ------");
	}
}
3.2.3 创建BeanPostProcessor实例
@Component
public class ScanAlarmBeanPostProcessor implements BeanPostProcessor, Ordered {
	private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());


	@Override
	public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {

		return bean;
	}

	/**
	 * 同上,不过时在初始化之后执行
	 * @param bean
	 * @param beanName
	 * @return
	 * @throws BeansException
	 */
	@Override
	public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
		// 后置的处理,我们不需要进行处理
		//目前只有MqSync类下DoSync注解需要读取配置
		Class<?> targetClass = bean.getClass();
		// logger.error("------ 扫描到带有 AlarmTarget 注解的 Bean: " + targetClass.getName() + " ------");
		AlarmTarget alarmTarget = AnnotationUtils.findAnnotation(targetClass, AlarmTarget.class);
		if (alarmTarget != null) {
			String className = bean.getClass().getName();
			logger.error("------ 扫描到带有 AlarmTarget 注解的 Bean: " + className + " ------");
			if (AopUtils.isCglibProxy(bean)) {
				className = className.substring(0, className.indexOf("$"));
			}
			logger.error("------ 扫描到带有 AlarmTarget 注解的 Bean: " + className + " ------");
			// 通过反射获取所有的方法
			Method[] methods = bean.getClass().getDeclaredMethods();
			for (Method method : methods) {
				logger.error("------ 扫描到带有 AlarmTarget 注解的 Bean 的方法: " + method.getName() + " ------");
				// 执行方法
				try {
					// 通过反射执行该类的所有的方法
					method.invoke(bean);
				} catch (Exception e) {
					logger.error("------ 执行方法出错! ------");
				}
			}
		}
		return bean;
	}

	@Override
	public int getOrder() {
		// 指定优先级
		return HIGHEST_PRECEDENCE + 1;
	}
}
3.2.4.测试
@SpringBootTest
class CustomAnnotationBeanPostProcessorTest {

    @Autowired
    private CustomerServiceInfo customerService;

    @Test
    void contextLoads() {
        // 测试类无需额外逻辑,只需观察控制台输出
    }
}

在这里插入图片描述

4. 优点与缺点

4.1 灵活性、可扩展性等优势分析

优点:

  • 灵活性: BeanPostProcessor 提供了一种灵活的机制,允许开发人员在 Spring 容器实例化、配置和初始化 Bean 的过程中介入,可以对 Bean 进行自定义的操作和处理。
  • 可扩展性: 开发人员可以根据实际需求编写自定义的 BeanPostProcessor 实现,实现各种功能,如自动装配、依赖注入、AOP、事务管理等。
  • AOP 实现基础: Spring AOP 的实现正是基于 BeanPostProcessor 机制。通过 BeanPostProcessor,Spring 可以在 Bean 初始化的过程中动态地生成代理对象,实现 AOP 的横切逻辑。

4.2 性能影响、复杂性等缺点讨论

缺点:

  • 性能影响: 使用 BeanPostProcessor 可能会对应用程序的性能产生一定的影响,特别是在大规模应用中,由于需要对所有的 Bean 进行处理,可能会增加应用程序的启动时间和内存消耗。
  • 复杂性: 如果不正确地使用 BeanPostProcessor,可能会导致应用程序的复杂性增加,增加代码的维护成本和理解难度。
  • 潜在的问题: 由于 BeanPostProcessor 可以对所有的 Bean 进行操作,因此可能会引入一些潜在的问题,如循环依赖、死锁等。

5. 使用注意事项

5.1 常见问题的解决方案

  • 避免循环依赖: 使用 BeanPostProcessor 时,要注意避免循环依赖的问题,避免出现 Bean 依赖循环导致的应用程序启动失败。
  • 谨慎处理 Bean 初始化逻辑: 在编写 BeanPostProcessor 实现时,要谨慎处理 Bean 初始化逻辑,确保不会影响到 Bean 的正常初始化过程。
  • 注意性能影响: 如果应用程序启动时间较长或者内存消耗较高,可以考虑减少 BeanPostProcessor 的数量,或者优化 BeanPostProcessor 的实现,以减少性能影响。
  • 优先级设置: 可以通过实现 Ordered 接口或者在 @Order 注解中设置优先级来控制多个 BeanPostProcessor 的执行顺序,确保处理顺序的正确性。
  • 测试与调试: 在使用 BeanPostProcessor 时,建议进行充分的测试和调试,确保 BeanPostProcessor 的实现逻辑正确,不会引入潜在的问题。

通过遵循上述注意事项,可以更加安全、高效地使用 BeanPostProcessor,并最大限度地发挥其在 Spring 应用程序中的作用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1469608.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux之项目部署与发布

目录 一、Nginx配置安装&#xff08;自启动&#xff09; 1.一键安装4个依赖 2. 下载并解压安装包 3. 安装Nginx 4. 启动 nginx 服务 5. 对外开放端口 6. 配置开机自启动 7.修改/etc/rc.d/rc.local的权限 二、后端部署tomcat负载均衡 1. 准备2个tomcat 2. 修改端口 3…

随机分布模型

目录 前言 一、离散型随机变量 1.1 0-1分布 1.2 二项分布 1.3 帕斯卡分布 1.4 几何分布 1.5 超几何分布 1.6 泊松分布 二、连续型随机变量 2.1 均匀分布 2.2 指数分布 2.3 高斯分布/正态分布 2.4 分布&#xff08;抽样分布&#xff09; 2.5 t分布&#xff08;抽样…

Vue局部注册组件实现组件化登录注册

Vue局部注册组件实现组件化登录注册 一、效果二、代码1、index.js2、App.vue3、首页4、登录&#xff08;注册同理&#xff09; 一、效果 注意我这里使用了element组件 二、代码 1、index.js import Vue from vue import VueRouter from vue-router import Login from ../vie…

迷你世界之建筑生成球体

local x0,y0,z00,30,0--起点坐标 local dx,dy,dz60,60,60--外切长方体横纵竖长度 local count,all0,dx*dy*dz--计数&#xff0c;总数 local m,k10000,0--单次生成方块数&#xff0c;无用循环值 local x,y,z0,0,0--当前坐标 local demath.random(2,19)/2 local id600--方块…

【监督学习之逻辑回归】

曾梦想执剑走天涯&#xff0c;我是程序猿【AK】 目录 简述概要知识图谱1.什么是逻辑回归&#xff1f;2.逻辑回归有哪些应用&#xff1f;3.回归分析如何工作&#xff1f;4.逻辑回归模型如何工作&#xff1f;5.逻辑回归分析有哪些类型&#xff1f;6.逻辑回归与其他机器学习技术相…

APP攻防-实战拿下某seseAPPSpringboot未授权HeapDump提取OSS利用

知识点 1、APK-抓包 2、资产信息收集 3、SpringBoot-漏洞利用 4、自动化工具 5、HeapDump-分析提取 6、AccessKEY-利用后续 演示案例&#xff1a; 1、APK-抓包 2、资产信息收集 3、SpringBoot-漏洞利用 SpringBoot漏洞利用&#xff1a; https://github.com/LandGrey/Spring…

K8S—集群调度

目录 前言 一 List-Watch 1.1 list-watch概述 1.2 list-watch工作机制 二 集群调度 2.1 调度过程 2.2 Predicate 和 Priorities 的常见算法和优先级选项 2.3 调度方式 三 亲和性 3.1 节点亲和性 3.2 Pod 亲和性 3.3 键值运算关系 3.4 Pod亲和性与反亲和性 3.5 示例…

【小记】简历 Tips

基本信息 真实诚信 教育经历&#xff08;院校信息&#xff09;、成绩、在校经历&#xff1b; 公司、项目经历 注意事项 逻辑清晰、重点突出、岗位JD完全吻合&#xff1b; 简洁&#xff08;一页纸简历、顺序从上到下&#xff09; 做了什么&#xff08;时间顺序&#xff09; 成果…

Escalate_Linux-环境变量劫持提权(5)

环境变量劫持提权 在Shll输入命令时&#xff0c;Shel会按PAH环境变量中的路径依次搜索命令&#xff0c;若是存在同名的命令&#xff0c;则执行最先找到的&#xff0c;若是PATH中加入了当前目录&#xff0c;也就是“”这个符号&#xff0c;则可能会被黑客利用&#xff0c;例如在…

冯诺依曼体系结构 计算机组成的金字塔

01 冯诺依曼体系结构&#xff1a;计算机组成的金字塔 学习计算机组成原理&#xff0c;到底是在学些什么呢&#xff1f;这个事儿&#xff0c;一两句话还真说不清楚。不过没关系&#xff0c;我们先从“装电脑”这个看起来没有什么技术含量的事情说起&#xff0c;来弄清楚计算机到…

Zookeeper客户端命令、JAVA API、监听原理、写数据原理以及案例

1. Zookeeper节点信息 指定服务端&#xff0c;启动客户端命令&#xff1a; bin/zkCli.sh -server 服务端主机名:端口号 1&#xff09;ls / 查看根节点下面的子节点 ls -s / 查看根节点下面的子节点以及根节点详细信息 其中&#xff0c;cZxid是创建节点的事务id&#xff0c…

QWidget: Must construct a QApplication before a QWidget 13:25:48: 程序异常结束。

QWidget: Must construct a QApplication before a QWidget 13:25:48: 程序异常结束。 你的插件是release&#xff0c;而你用了debug模式、

Yolov8有效涨点:YOLOv8-AM,采用多种注意力模块提高检测精度,含代码,超详细

前言 2023 年&#xff0c;Ultralytics 推出了最新版本的 YOLO 模型。注意力机制是提高模型性能最热门的方法之一。 本次介绍的是YOLOv8-AM&#xff0c;它将注意力机制融入到原始的YOLOv8架构中。具体来说&#xff0c;我们分别采用四个注意力模块&#xff1a;卷积块注意力模块…

max_element和min_element使用

头文件 #include<alorithm> 作用 用于返回数组或容器中最值元素(最小值、最大值)&#xff0c;值和下标。 使用举例 #include<iostream> #include<vector> #include<algorithm> using namespace std; int main() {/*数组初始化*/vector<int>…

Django入门指南:从环境搭建到模型管理系统的完整教程

环境安装&#xff1a; ​ 由于我的C的Anaconda 是安装在C盘的&#xff0c;但是没内存了&#xff0c;所有我将环境转在e盘&#xff0c;下面的命令是创建环境到指定目录中. conda create --prefixE:\envs\dj42 python3.9进入环境中&#xff1a; conda activate E:\envs\dj42…

unity学习(41)——创建(create)角色脚本(panel)——UserHandler(收)+CreateClick(发)——发包!

1.客户端的程序结构被我精简过&#xff0c;现在去MessageManager.cs中增加一个UserHandler函数&#xff0c;根据收到的包做对应的GameInfo赋值。 2.在Model文件夹下新增一个协议文件UserProtocol&#xff0c;内容很简单。 using System;public class UserProtocol {public co…

Unity与Android交互通信系列(5)

在前述文章中&#xff0c;已经使用了AndroidJavaProxy代理接口&#xff0c;本节我们将详细的介绍AndroidJavaProxy代理的用法。正如其名&#xff0c;AndroidJavaProxy是一个代理&#xff0c;它在Android端代码与Unity端代码交互中起一个桥接作用。其一般用法为在Java代码中定义…

【C++】类和对象之拷贝构造函数篇

个人主页 &#xff1a; zxctscl 文章封面来自&#xff1a;艺术家–贤海林 如有转载请先通知 文章目录 1. 前言2. 传值传参和传引用传参3. 概念4. 特征 1. 前言 在前面学习了6个默认成员函数中的构造函数和析构函数 【C】构造函数和析构函数详解&#xff0c;接下来继续往后看拷…

S-35390A计时芯片介绍及开发方案

计时芯片 S-35390A芯片是计时芯片&#xff0c;一般用来计算时间。低功耗&#xff0c;宽电压&#xff0c;受温度影响小&#xff0c;适用于很多电路。它有一个问题&#xff0c;不阻止用户设置不存在的时间&#xff0c;设置进去之后计时或者闹钟定时会出错。 规格书阅读 首先我…

如何移除禁用WordPress默认小工具(附WordPress默认小工具名称)

WordPress 自带的小工具非常多&#xff0c;但是我们用到的也就那么几种&#xff0c;甚至一种都不会用到&#xff0c;所以很有必要注销&#xff08;去除&#xff09;掉一些不用的小工具。实现的方法也很简单&#xff0c;只需将以下代码&#xff0c;根据自己的情况删除需要用的小…