Spring 源码解读:自定义实现BeanPostProcessor的扩展点
引言
在Spring的生命周期管理中,BeanPostProcessor是一个非常重要的扩展点。它允许开发者在Bean初始化的前后插入自定义的逻辑,从而实现更灵活的Bean管理。BeanPostProcessor是Spring框架中用于对Bean实例进行修改的机制之一。通过实现该接口,开发者可以在Bean创建过程中添加额外的逻辑。在本篇文章中,我们将手动实现一个类似于Spring的BeanPostProcessor,展示如何在Bean初始化的前后进行扩展处理,并与Spring的BeanPostProcessor机制进行对比。
摘要
Spring的BeanPostProcessor是一个用于在Bean初始化前后进行处理的扩展点。本文将通过手动实现一个简化版的BeanPostProcessor,展示如何利用它在Bean生命周期的不同阶段插入自定义逻辑。我们还将与Spring中的BeanPostProcessor机制进行对比,帮助读者理解扩展点的工作原理及其在实际项目中的应用。
什么是BeanPostProcessor
在Spring中,BeanPostProcessor是一个允许在Bean初始化前后执行额外逻辑的接口。它提供了两个主要方法:
postProcessBeforeInitialization():在Bean的@PostConstruct或初始化方法之前调用。postProcessAfterInitialization():在Bean的初始化方法之后调用。
通过BeanPostProcessor,开发者可以在Bean的整个生命周期中,注入额外的行为。例如,开发者可以使用它实现AOP功能、Bean属性修改、注解处理等。
Spring中的BeanPostProcessor接口
Spring中的BeanPostProcessor接口定义如下:
public interface BeanPostProcessor {default Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {return bean;}default Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {return bean;}
}
postProcessBeforeInitialization():在Bean的初始化之前调用。postProcessAfterInitialization():在Bean的初始化之后调用。
接下来,我们将手动实现一个简化版的BeanPostProcessor,展示如何利用它扩展Bean的生命周期处理。
手动实现BeanPostProcessor扩展点
为了更好地理解BeanPostProcessor的设计原理和应用场景,我们将通过一个简化的自定义实现,演示如何在Bean初始化的前后执行额外的逻辑。
步骤概述
- 定义
BeanPostProcessor接口:提供Bean初始化前后处理的扩展接口。 - 实现
BeanPostProcessor接口:定义前后处理逻辑。 - 实现Bean工厂类:在Bean创建时调用
BeanPostProcessor。 - 测试自定义
BeanPostProcessor机制:验证扩展点的工作流程。
定义BeanPostProcessor接口
首先,我们定义一个类似于Spring的BeanPostProcessor接口。它包含两个方法,分别在Bean初始化之前和之后调用。
/*** 自定义BeanPostProcessor接口,用于在Bean初始化前后执行自定义处理*/
public interface BeanPostProcessor {/*** 在Bean初始化之前执行* @param bean 目标Bean实例* @param beanName Bean的名称* @return 处理后的Bean*/Object postProcessBeforeInitialization(Object bean, String beanName);/*** 在Bean初始化之后执行* @param bean 目标Bean实例* @param beanName Bean的名称* @return 处理后的Bean*/Object postProcessAfterInitialization(Object bean, String beanName);
}
实现自定义BeanPostProcessor
接下来,我们实现一个具体的BeanPostProcessor,在Bean初始化的前后打印日志信息。
/*** 自定义的BeanPostProcessor实现,打印Bean的初始化过程*/
public class CustomBeanPostProcessor implements BeanPostProcessor {@Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) {System.out.println("Before Initialization of Bean: " + beanName);return bean; // 返回原始Bean实例}@Overridepublic Object postProcessAfterInitialization(Object bean, String beanName) {System.out.println("After Initialization of Bean: " + beanName);return bean; // 返回原始Bean实例}
}
说明:
postProcessBeforeInitialization()方法在Bean初始化之前被调用,输出日志信息。postProcessAfterInitialization()方法在Bean初始化之后被调用,输出日志信息。
实现Bean工厂类
为了支持BeanPostProcessor的调用,我们将扩展Bean工厂类,在创建Bean时调用BeanPostProcessor。
import java.util.ArrayList;
import java.util.List;/*** 简单的Bean工厂类,支持注册BeanPostProcessor*/
public class SimpleBeanFactory {private List<BeanPostProcessor> postProcessors = new ArrayList<>();/*** 注册BeanPostProcessor* @param processor BeanPostProcessor实现*/public void addPostProcessor(BeanPostProcessor processor) {postProcessors.add(processor);}/*** 创建Bean实例,并调用BeanPostProcessor* @param clazz Bean的Class类型* @return 创建的Bean实例*/public Object createBean(Class<?> clazz) throws Exception {String beanName = clazz.getSimpleName();Object bean = clazz.getDeclaredConstructor().newInstance();// 在Bean初始化之前执行所有的BeanPostProcessorfor (BeanPostProcessor processor : postProcessors) {bean = processor.postProcessBeforeInitialization(bean, beanName);}// 调用Bean的初始化方法(此处省略初始化逻辑)// 在Bean初始化之后执行所有的BeanPostProcessorfor (BeanPostProcessor processor : postProcessors) {bean = processor.postProcessAfterInitialization(bean, beanName);}return bean;}
}
说明:
SimpleBeanFactory支持BeanPostProcessor的注册,并在Bean初始化的前后调用它们。- 在
createBean()方法中,Bean的初始化过程被扩展为包括前后处理逻辑。
实现测试类
接下来,通过一个简单的测试类,验证自定义的BeanPostProcessor扩展点机制。
/*** 测试自定义的BeanPostProcessor扩展点*/
public class BeanPostProcessorTest {public static void main(String[] args) throws Exception {// 创建Bean工厂SimpleBeanFactory beanFactory = new SimpleBeanFactory();// 注册自定义的BeanPostProcessorbeanFactory.addPostProcessor(new CustomBeanPostProcessor());// 创建BeanObject myBean = beanFactory.createBean(MyBean.class);}
}/*** 一个简单的Bean类*/
public class MyBean {public MyBean() {System.out.println("MyBean constructor called");}public void initialize() {System.out.println("MyBean initialization method called");}
}
测试结果:
- 当
MyBean被创建时,BeanPostProcessor的前后处理逻辑都会被调用,输出对应的日志信息。
输出:
Before Initialization of Bean: MyBean
MyBean constructor called
After Initialization of Bean: MyBean
类图与流程图
为了更好地理解BeanPostProcessor扩展点的工作原理,我们提供了类图和流程图。
类图
流程图
Spring中的BeanPostProcessor解析
在Spring中,BeanPostProcessor是一个核心扩展点,
允许开发者在Bean初始化的前后执行自定义逻辑。这使得开发者可以在Bean的创建过程中,灵活地插入额外的行为,例如属性注入、代理生成等。
Spring中的典型用法
- AOP实现:Spring中的AOP功能就是通过
BeanPostProcessor来实现的。在Bean初始化之后,生成代理对象并返回代理Bean。 - 属性注入:Spring的
@Autowired注解依赖于BeanPostProcessor来完成依赖注入。 - 生命周期管理:Spring可以在Bean的生命周期中插入自定义的行为,例如初始化前后的额外处理。
Spring的BeanPostProcessor源码解析
Spring在AbstractAutowireCapableBeanFactory中通过如下代码来执行BeanPostProcessor的处理逻辑:
protected Object initializeBean(final String beanName, final Object bean, @Nullable RootBeanDefinition mbd) {Object wrappedBean = bean;// 在Bean初始化之前执行BeanPostProcessorwrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);// 执行初始化逻辑invokeInitMethods(beanName, wrappedBean, mbd);// 在Bean初始化之后执行BeanPostProcessorwrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);return wrappedBean;
}
解析:
- Spring通过
applyBeanPostProcessorsBeforeInitialization()和applyBeanPostProcessorsAfterInitialization()来调用所有的BeanPostProcessor,以在Bean初始化的前后执行扩展处理。
对比分析:手动实现与Spring的区别
-
功能复杂度:
- Spring:Spring的
BeanPostProcessor支持更多高级功能,如AOP、依赖注入等,并且可以在不同阶段插入扩展逻辑。 - 简化实现:我们的自定义实现展示了
BeanPostProcessor的基本工作原理,但缺少Spring中的高级功能。
- Spring:Spring的
-
扩展性:
- Spring:Spring的
BeanPostProcessor可以与其他Spring特性无缝集成,如事务管理、AOP等,提供更强的扩展能力。 - 简化实现:我们实现的版本主要用于演示基本原理,未提供丰富的扩展机制。
- Spring:Spring的
-
集成能力:
- Spring:
BeanPostProcessor可以与Spring容器中的其他扩展点(如BeanFactoryPostProcessor、@Autowired)一起工作,形成一个强大的Bean管理机制。 - 简化实现:我们的实现是独立的,不具备与其他框架组件的集成能力。
- Spring:
总结
通过手动实现一个BeanPostProcessor扩展点,我们展示了如何在Bean初始化的前后执行自定义逻辑。这种扩展机制在Spring中被广泛应用于AOP、依赖注入和生命周期管理。Spring中的BeanPostProcessor为开发者提供了强大的工具,帮助在Bean的创建过程中灵活插入自定义逻辑。理解这一机制将有助于您在实际项目中更好地管理Bean的生命周期和扩展功能。
互动与思考
你在项目中是否遇到过需要在Bean初始化前后执行自定义逻辑的场景?你认为BeanPostProcessor在哪些场景下最为有用?欢迎在评论区分享你的经验与见解!
如果你觉得这篇文章对你有帮助,请别忘了:
- 点赞 ⭐
- 收藏 📁
- 关注 👀
让我们一起深入学习Spring框架,成为更优秀的开发者!
相关文章:
Spring 源码解读:自定义实现BeanPostProcessor的扩展点
引言 在Spring的生命周期管理中,BeanPostProcessor是一个非常重要的扩展点。它允许开发者在Bean初始化的前后插入自定义的逻辑,从而实现更灵活的Bean管理。BeanPostProcessor是Spring框架中用于对Bean实例进行修改的机制之一。通过实现该接口࿰…...
Spring Boot-分布式系统问题
Spring Boot 在分布式系统中的常见问题及解决方案 随着互联网的发展,系统规模和复杂度越来越大,分布式系统成为应对高并发、大数据量场景的重要架构选择。Spring Boot 作为一种轻量级的开发框架,广泛应用于构建微服务和分布式系统中。然而&a…...
面试题总结(三) -- 内存管理篇
面试题总结(三) – 内存管理篇 文章目录 面试题总结(三) -- 内存管理篇<1> C 中堆内存和栈内存的区别是什么?<2> 如何在 C 中手动管理内存(new/delete 操作符)?<3> C 中内存泄漏的原因和避免方法<4> 谈谈…...
Qt 定时器-定时备份
定时备份 在Qt 中,可以使用QTimer类来实现定时备份功能。以下是一个示例代码,每隔一段时间自动执行备份操作: #include <QTimer>QTimer timer; int backupInterval 24 * 60 * 60 * 1000;//备份间隔为24小时connect(&timer, &…...
天融信把桌面explorer.exe删了,导致开机之后无windows桌面,只能看到鼠标解决方法
win10开机进入桌面,发现桌面无了,但是可以ctrlaltdelete调出任务管理器 用管理员权限打开cmd,输入: sfc /scanfilec:\windowslexplorer.exe 在运行C:\windows\Explorer.exe;可以进入桌面,但是隔离几秒钟…...
视频分割操作教程
1、打开剪映 2、点击开始创作上面的“”,选择视频,点击添加按钮,导入一个视频素材到剪映 3、滑动视频,让视频竖线到合适位置 4、点击视频,出现白色边框 5、点击工具栏“分割”,然后点击需要删除的视频部分 …...
唯品会大数据面试题及参考答案(3万字长文)
synchronized 和 volatile 的区别 synchronized是 Java 中的关键字,用于实现同步机制,确保在同一时刻只有一个线程可以访问被它修饰的代码块或方法。volatile也是 Java 中的关键字,主要用于保证变量的可见性。 功能方面: synchronized可以保证原子性、可见性和有序性。它通…...
使用容器技术快速入门MinIO
使用容器技术快速入门MinIO 使用容器技术(docker或者podman)快速部署一个单节点单磁盘 MinIO 服务器,用于对MinIO对象存储及其兼容 S3 的 API 层进行早期的开发和评估。 1. 准备工作 机器已经安装了 Podman 或者 Docker 。 对用于持久卷的…...
ros2教程(一):使用python和C++发布摄像头原始图像和压缩图像
1. 使用python发布图像 在ROS 2中,可以通过使用rclpy库来发布压缩图像和原始图像。发布原始图像可以使用sensor_msgs.msg.Image消息类型,压缩图像则使用sensor_msgs.msg.CompressedImage消息类型。 #!/usr/bin/env python3# function: usbcam publish r…...
【自动化测试】UI自动化的分类、如何选择合适的自动化测试工具以及其中appium的设计理念、引擎和引擎如何工作
引言 UI自动化测试主要针对软件的用户界面进行测试,以确保用户界面元素的交互和功能符合预期 文章目录 引言一、UI自动化的分类1.1 基于代码的自动化测试1.2 基于录制/回放的自动化测试1.3 基于框架的自动化测试1.4 按测试对象分类1.5 按测试层次分类1.6 按测试执行…...
深入理解Python中的“_,”:一个实用的语法特性
在Python编程中,你可能经常会看到一个特殊的标识符“_”。这个符号在Python中有多种用途,其具体含义依上下文而定。本文将探讨其中一种常见用法——作为一个临时性的占位符——并解释它在实际编程中的实用性和应用场景。 1. “_”作为占位符 在Python中…...
Mac清理其他文件:释放存储空间的高效指南
每个Mac用户都可能遇到存储空间不足的问题,尤其是当“其他”文件积累到一定体积时。在Mac上,“其他”文件通常包括各种系统文件、缓存、文档以及不被归类为应用程序、照片、电影或音乐的其他类型的文件。这些文件往往不易被注意,但逐渐占用了…...
html+css+js网页设计 旅游 龙门石窟4个页面
htmlcssjs网页设计 旅游 龙门石窟4个页面 网页作品代码简单,可使用任意HTML辑软件(如:Dreamweaver、HBuilder、Vscode 、Sublime 、Webstorm、Text 、Notepad 等任意html编辑软件进行运行及修改编辑等操作)。 获取源码 1&#…...
CISSP一站通关
依托轻速云维护了一个专注于CISSP备考通关的在线学习平台,提供知识串讲视频,配合大量针对性的习题和重难点习题解析,帮助备考学习者高效学习和巩固知识点。已经帮助100考友顺利通过考试。 知识串讲视频是我主讲的5天直播课程的录屏࿰…...
Golang | Leetcode Golang题解之第406题根据身高重建队列
题目: 题解: func reconstructQueue(people [][]int) (ans [][]int) {sort.Slice(people, func(i, j int) bool {a, b : people[i], people[j]return a[0] > b[0] || a[0] b[0] && a[1] < b[1]})for _, person : range people {idx : pe…...
【我的Android进阶之旅】解决CardView四个圆角有白边的问题
文章目录 一、问题描述二、分析CardView出现白边的原因三、如何解决这个问题?3.1 如何修复?3.2 为什么这样可以修复?3.3 示例代码3.4 总结一、问题描述 在实现一个RecycleView的Item时候,样式需要用到卡片式效果,于是想到用CardView来实现,但是最终发现运行出来的效果,…...
学习笔记JVM篇(四)
垃圾回收器 说完垃圾回收算法接下来就需要对应的垃圾回收器去回垃圾回收器。接下来介绍几种垃圾回收器 1、Serial 串行回收器,是单线程版本,暂停所有的应用。在单CPU的情况下效率是很高的,因为不涉及线程的上下文切换。适用于小型程序和客…...
828 华为云征文|华为 Flexus 云服务器搭建萤火商城 2.0
《828 华为云征文|华为 Flexus 云服务器搭建萤火商城 2.0》 在 2024 年 9 月 14 日这个特别的日子,我着手利用华为 Flexus 云服务器搭建轻量级、高性能、前后端分离的电商系统萤火商城 2.0,开启了一段充满挑战与惊喜的旅程。 华为 Flexus 云服…...
centos7安装MySQL5.7.44
下载压缩文件 命令: #放到在/usr/local目录下 cd /usr/local #上传命令选择安装包 rz #解压缩包 tar -zxvf mysql-5.7.44-linux-glibc2.12-x86_64.tar.gz #给包重命名为mysql mv mysql-5.7.44-linux-glibc2.12-x86_64 mysql #查看mysql目录下有什么东西 [rootlocal…...
HTTP 请求处理的完整流程到Servlet流程图
HTTP 请求处理的完整流程。从 TCP 三次握手开始,一直到 Servlet 处理请求并返回响应。 首先,让我解释一下 response.setContentType("text/html;charsetUTF-8"); 这行代码: 这行代码设置了 HTTP 响应的 Content-Type 头。它告诉浏…...
生成xcframework
打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...
Qt/C++开发监控GB28181系统/取流协议/同时支持udp/tcp被动/tcp主动
一、前言说明 在2011版本的gb28181协议中,拉取视频流只要求udp方式,从2016开始要求新增支持tcp被动和tcp主动两种方式,udp理论上会丢包的,所以实际使用过程可能会出现画面花屏的情况,而tcp肯定不丢包,起码…...
简易版抽奖活动的设计技术方案
1.前言 本技术方案旨在设计一套完整且可靠的抽奖活动逻辑,确保抽奖活动能够公平、公正、公开地进行,同时满足高并发访问、数据安全存储与高效处理等需求,为用户提供流畅的抽奖体验,助力业务顺利开展。本方案将涵盖抽奖活动的整体架构设计、核心流程逻辑、关键功能实现以及…...
通过Wrangler CLI在worker中创建数据库和表
官方使用文档:Getting started Cloudflare D1 docs 创建数据库 在命令行中执行完成之后,会在本地和远程创建数据库: npx wranglerlatest d1 create prod-d1-tutorial 在cf中就可以看到数据库: 现在,您的Cloudfla…...
【Linux】C语言执行shell指令
在C语言中执行Shell指令 在C语言中,有几种方法可以执行Shell指令: 1. 使用system()函数 这是最简单的方法,包含在stdlib.h头文件中: #include <stdlib.h>int main() {system("ls -l"); // 执行ls -l命令retu…...
SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...
Leetcode 3577. Count the Number of Computer Unlocking Permutations
Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接:3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯,要想要能够将所有的电脑解锁&#x…...
今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存
文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...
初学 pytest 记录
安装 pip install pytest用例可以是函数也可以是类中的方法 def test_func():print()class TestAdd: # def __init__(self): 在 pytest 中不可以使用__init__方法 # self.cc 12345 pytest.mark.api def test_str(self):res add(1, 2)assert res 12def test_int(self):r…...
使用LangGraph和LangSmith构建多智能体人工智能系统
现在,通过组合几个较小的子智能体来创建一个强大的人工智能智能体正成为一种趋势。但这也带来了一些挑战,比如减少幻觉、管理对话流程、在测试期间留意智能体的工作方式、允许人工介入以及评估其性能。你需要进行大量的反复试验。 在这篇博客〔原作者&a…...
