【Spring源码解读!底层原理进阶】【下】探寻Spring内部:BeanFactory和ApplicationContext实现原理揭秘✨

 🎉🎉欢迎光临🎉🎉

🏅我是苏泽,一位对技术充满热情的探索者和分享者。🚀🚀

🌟特别推荐给大家我的最新专栏《Spring 狂野之旅:底层原理高级进阶》 🚀

本专栏纯属为爱发电永久免费!!!

这是苏泽的个人主页可以看到我其他的内容哦👇👇

努力的苏泽icon-default.png?t=N7T8http://suzee.blog.csdn.net/


 

目录

深挖ApplicationContext的高级特性

环境与配置文件的灵活管理

Profile的工作原理

使用PropertySources优雅地管理配置

数据访问与事务管理的抽象

数据访问异常的统一处理

声明式事务管理的实现机制

实践:验证声明式事务的工作原理

第五章:Spring表达式语言(SpEL)

SpEL的设计目的与应用场景

SpEL的核心语法与功能

如何通过SpEL实现动态配置

总结:BeanFactory与ApplicationContext的精髓

如何继续深入学习Spring


深挖ApplicationContext的高级特性

在Spring框架中,ApplicationContext 被誉为Spring的心脏,负责管理Bean的生命周期和提供配置框架的各种高级特性。本篇博客将深入探讨ApplicationContext的几个高级特性,包括环境与配置文件的灵活管理、Profile的工作原理、使用PropertySources管理配置的优雅方式,以及数据访问与事务管理的抽象和实现机制。我们不仅会通过源码解读这些特性背后的设计思想,还会提供代码示例来验证我们的观点。

环境与配置文件的灵活管理

Spring允许开发者通过多种方式灵活管理应用的配置,包括但不限于属性文件、YAML文件、环境变量和命令行参数。这一切得益于EnvironmentPropertySources抽象。

Profile的工作原理

Profile允许开发者为不同的环境(如开发、测试、生产)定义不同的配置。通过激活特定的Profile,可以加载相应环境的配置。

@Configuration
@Profile("dev")
public class DevConfig {// 配置类内容
}

上述代码展示了如何定义一个仅在开发环境下激活的配置类。Spring根据当前激活的Profile来决定是否加载该配置类。

使用PropertySources优雅地管理配置

PropertySources是Spring环境抽象的一部分,它允许开发者从多个来源灵活地加载配置

Environment env = applicationContext.getEnvironment();
String property = env.getProperty("some.property");

通过Environment接口,可以方便地访问配置属性。

数据访问与事务管理的抽象

Spring提供了一套与具体技术无关的数据访问异常层次结构,使得异常处理更加统一和简便。

数据访问异常的统一处理

Spring将底层数据访问技术(如JDBC、Hibernate等)抛出的异常转换为DataAccessException体系中的异常,从而避免了与特定技术的耦合。

声明式事务管理的实现机制

Spring的声明式事务管理依赖于AOP(面向切面编程),允许开发者通过声明的方式来管理事务,而无需编写传统的事务管理代码。

@Transactional
public void someTransactionalMethod() {// 方法体
}

上述@Transactional注解表明该方法应在事务的上下文中运行。Spring在幕后自动创建和管理事务。

实践:验证声明式事务的工作原理

为了进一步理解声明式事务的工作原理,我们可以编写一个简单的测试用例来验证事务的行为。

 

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = {TransactionConfig.class})
public class TransactionalTest {@Autowiredprivate TestService testService;@Test(expected = DataAccessException.class)public void testTransactionalMethod() {testService.someTransactionalMethod();}
}

在这个测试中,someTransactionalMethod方法在遇到数据访问异常时能够回滚事务。

如果事务正确回滚,那么这个测试应该通过,因为我们期望遇到DataAccessException

为了演示和验证事务的回滚机制,我们可以通过一个简单的Spring Boot应用中的服务层方法来模拟。假设我们有一个UserService,它负责处理用户的注册逻辑。在用户注册的过程中,我们故意引入一个数据访问异常,以触发事务回滚。

首先,我们定义一个简单的用户实体User和对应的数据访问接口UserRepository(这里只是为了演示,不涉及具体的数据库操作代码):

@Entity
public class User {@Id@GeneratedValue(strategy = GenerationType.AUTO)private Long id;private String username;// 省略构造函数、Getter和Setter方法
}public interface UserRepository extends JpaRepository<User, Long> {// 这里可以添加一些自定义的数据访问方法
}

接下来,我们实现UserService,并在其中添加一个注册用户的方法,该方法会故意抛出一个DataAccessException来模拟数据访问异常:

@Service
public class UserService {@Autowiredprivate UserRepository userRepository;@Transactionalpublic void registerUser(User user) {// 正常的用户注册逻辑userRepository.save(user);System.out.println("用户注册成功");// 故意抛出数据访问异常来模拟异常情况if (user.getUsername().equals("triggerException")) {throw new DataAccessException("模拟数据访问异常") {};}}
}

最后,我们编写一个测试用例来验证当registerUser方法遇到数据访问异常时,事务是否能够正确回滚:

@RunWith(SpringRunner.class)
@SpringBootTest
public class UserServiceTest {@Autowiredprivate UserService userService;@Autowiredprivate UserRepository userRepository;@Test(expected = DataAccessException.class)public void testRegisterUserWithException() {User user = new User();user.setUsername("triggerException");try {userService.registerUser(user);} finally {// 验证用户没有被保存到数据库,即事务回滚了Optional<User> foundUser = userRepository.findByUsername("triggerException");assertFalse(foundUser.isPresent());System.out.println("事务回滚,用户未被保存到数据库");}}
}

我们首先尝试注册一个用户名为triggerException的用户,这将触发我们在registerUser方法中故意设置的数据访问异常。根据@Transactional注解的工作原理,一旦方法抛出异常,所有的数据变更都应该被回滚。

最后在捕获异常后,我们检查数据库是否存在该用户记录,找不到的,这样就证明事务确实被回滚了

第五章:Spring表达式语言(SpEL)

在本章中,我们将探讨Spring表达式语言(SpEL)的设计目的、应用场景以及它的核心语法与功能。SpEL是一个强大的表达式语言,它可以在Spring框架中被广泛地应用于动态配置和表达式求值的场景。

SpEL的设计目的与应用场景

首先,让我们来了解一下SpEL的设计目的和适用场景。SpEL的主要设计目的是为了提供一个灵活而强大的表达式语言,使得Spring框架能够更好地支持动态配置和运行时求值的需求。

对于应用场景来说,SpEL可以被广泛地应用于以下方面:

  • 动态配置:SpEL可以通过表达式来动态地配置Spring中的bean属性、方法参数等,从而实现更加灵活的配置方式。

  • 运行时求值:SpEL可以在运行时对表达式进行求值,从而实现动态计算、判断和决策等功能。

SpEL的核心语法与功能

现在让我们深入研究一下SpEL的核心语法和功能。SpEL的语法结构类似于Java,但也引入了一些新的概念和符号,使得表达式更加灵活和强大。

  • 字面量表达式:SpEL支持各种类型的字面量,包括字符串、数字、布尔值等。例如,'Hello, SpEL!'表示一个字符串字面量。

  • 属性访问:使用.操作符可以访问对象的属性。例如,person.name表示访问person对象的name属性。

  • 方法调用:使用.或者[]操作符可以调用对象的方法。例如,person.getName()表示调用person对象的getName()方法。

  • 运算符:SpEL支持各种运算符,包括算术运算符、关系运算符、逻辑运算符等。例如,1 + 2表示加法运算。

  • 条件表达式:SpEL支持使用三元运算符?:进行条件判断。例如,age >= 18 ? '成年人' : '未成年人'表示根据age的值判断是否成年。

  • 集合操作:SpEL支持对集合进行操作,包括访问集合元素、过滤、投影等。例如,numbers.![#this * 2]表示将numbers集合中的每个元素乘以2。

如何通过SpEL实现动态配置

现在让我们看一个具体的示例,来说明如何通过SpEL实现动态配置。假设我们有一个简单的Java类Person,它有一个名为age的属性。

public class Person {private int age;public int getAge() {return age;}public void setAge(int age) {this.age = age;}
}

我们可以通过SpEL来动态地配置Person对象的age属性。在Spring的配置文件中,使用#{}包裹SpEL表达式。

<bean id="person" class="com.example.Person"><property name="age" value="#{ 18 + 2 }" />
</bean>

上述配置中,SpEL表达式18 + 2会在运行时求值,并将结果赋值给Person对象的age属性。假设我们创建了一个person对象并获取其年龄:

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
Person person = context.getBean("person", Person.class);
System.out.println(person.getAge());

输出结果为:20

通过这个示例,我们可以看到如何使用SpEL来实现动态配置,通过表达式来计算属性的值,使得配置更加灵活和可变。

总结:BeanFactory与ApplicationContext的精髓

BeanFactoryApplicationContext是Spring容器的核心。BeanFactory提供了高级IoC的配置机制,而ApplicationContext在此基础上添加了更多企业所需的功能,如事件发布、国际化消息支持等。AOP正是ApplicationContext提供的众多高级特性之一,通过它,我们能够以简洁的方式实现应用中的横切关注点。

如何继续深入学习Spring

深入学习Spring的最佳方式是通过实践。建议读者不仅要阅读官方文档,还应该关注Spring的新特性和最佳实践。同时,参与开源项目、阅读源码、编写自己的Spring应用,都是提升自己技术水平的有效途径。

希望这篇博客能够帮助你更好地理解Spring中的AOP特性,以及ApplicationContext的强大功能。记住,学习之路是永无止境的,让我们一起在Spring的世界里不断探索,不断前进。

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

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

相关文章

[超分辨率重建]ESRGAN算法训练自己的数据集过程

一、下载数据集及项目包 1. 数据集 1.1 文件夹框架的介绍&#xff0c;如下图所示&#xff1a;主要有train和val&#xff0c;分别有高清&#xff08;HR&#xff09;和低清&#xff08;LR&#xff09;的图像。 1.2 原图先通过分割尺寸的脚本先将数据集图片处理成两个相同的图像…

c++之说_13|模板 折叠表达式

折叠表达式 可以通过形参包的的实际参数&#xff08;不是类型&#xff09; 展开式子 这是这里说的几种 实际上并还有一些写法 先介绍这几种吧 #include <cstdio> template<typename T,T... n> struct integer_sequence {T val; }; template<int idx,typenam…

GEE详细教程之:将Landsat8与Landsat9影像合成一个影像

1.前言 因项目需求&#xff0c;需要获取一个研究区的Landsat8影像&#xff0c;但Landsat8重复周期长&#xff0c;加之天气的影响&#xff0c;很难获取影像质量较好的影像。Landsat4/5/7的波段顺序与landsat8不同&#xff0c;除此之外&#xff0c;landsat7影像还需要工具进行条带…

【PyQt】08 - 编辑Tab顺序

文章目录 前言一、Tab顺序二、编辑Tab顺序总结 前言 介绍了什么是Tab顺序&#xff0c;以及如何修改Tab顺序。 一、Tab顺序 当你的界面设计好之后&#xff0c;在输入栏按住Tab按键&#xff0c;他会按照你摆放的顺序一次转跳 二、编辑Tab顺序 方法一 然后鼠标左击就可以改变…

阿里云参编业内首个代码大模型标准丨云原生 2024 年 1 月产品技术动态

云原生月度动态 云原生是企业数字创新的最短路径。 《阿里云云原生每月动态》&#xff0c;从趋势热点、产品新功能、服务客户、开源与开发者动态等方面&#xff0c;为企业提供数字化的路径与指南。 趋势热点 &#x1f947; 阿里云参编业内首个代码大模型标准&#xff0c;通…

LLaVA-1.6:多模态AI新标准,中文零样本能力与低成本训练革命,性能全面超越Gemini Pro

引言 2023年10月&#xff0c;LLaVA-1.5凭借其简洁高效的设计和在12个数据集上的出色表现&#xff0c;为大规模多模态模型&#xff08;LMM&#xff09;的研究和应用奠定了基础。进入2024年&#xff0c;我们迎来了LLaVA-1.6&#xff0c;一个在理性推理、光学字符识别&#xff08…

LeetCode1365之切披萨的方案数(相关话题:二维前缀和,动态规划)

题目描述 给你一个 rows x cols 大小的矩形披萨和一个整数 k &#xff0c;矩形包含两种字符&#xff1a; A &#xff08;表示苹果&#xff09;和 . &#xff08;表示空白格子&#xff09;。你需要切披萨 k-1 次&#xff0c;得到 k 块披萨并送给别人。 切披萨的每一刀&#xf…

Bpmn-js自定义Palette元素

Bpmn-js作为一个流程编辑器&#xff0c;常规的我们可以将其划分为几个功能区域&#xff0c;每个区域对应的负责不同的功能实现&#xff0c;bpmn-js的设计给我们留下了大量的留白和可扩展区域&#xff0c;其每一部分都可进行组合拼装&#xff0c;同时也支持我们的各种不同层次需…

『运维备忘录』之 Kubernetes(K8S) 常用命令速查

一、简介 kubernetes&#xff0c;简称K8s&#xff0c;是用8代替名字中间的8个字符“ubernete”而成的缩写&#xff0c;是一个开源的&#xff0c;用于管理云平台中多个主机上的容器化的应用。kubernetes是基于容器技术的分布式架构解决方案&#xff0c;具有完备的集群管理能力&a…

霍金《时间简史》(A Brief History of Time)学习笔记(第四章)

Chapter 4: The Uncertainty Principle Footnote: Chapter 4. Mainly talks about Werner Heisenberg’s Uncertainty Principle. Vital principle in modern physics, concept not hard to understand——work of a genius mind. Footnote: Werner Heisenberg, German physici…

【蓝桥杯冲冲冲】Invasion of the Milkweed G

【蓝桥杯冲冲冲】Invasion of the Milkweed G 蓝桥杯备赛 | 洛谷做题打卡day30 文章目录 蓝桥杯备赛 | 洛谷做题打卡day30[USACO09OCT] Invasion of the Milkweed G题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 题解代码我的一些话 [USACO09OCT] Invasion of the Mi…

斯巴鲁Subaru EDI需求分析

斯巴鲁Subaru是日本运输集团斯巴鲁公司&#xff08;前身为富士重工&#xff09;的汽车制造部门&#xff0c;以性能而闻名&#xff0c;曾赢得 3 次世界拉力锦标赛和 10 次澳大利亚拉力锦标赛。 斯巴鲁Subaru EDI 需求分析 企业与斯巴鲁Subaru建立EDI连接&#xff0c;首先需要确…

洛谷C++简单题小练习day9—[AHOI2017]寻找探监点

day9--[AHOI2017]寻找探监点--2.7 习题概述 题目描述 一个nn 的网格图&#xff08;标号由 1,1 开始&#xff09;上有 m 个探测器&#xff0c;每个探测器有个探测半径 r &#xff0c;问这 nn 个点中有多少个点能被探测到。 输入格式 第一行 3 个整数 n,m,r。 接下来 m 行&…

解决dockor安装nginx提示missing signature key的问题

问题描述 使用dockor安装nginx拉取nginx的时候提示key丢失问题 问题定位 由于dockor版本低导致 问题解决 卸载重新安装最新版本dockor 解决步骤 1. 卸载旧版本的Docker&#xff1a; sudo yum remove docker docker-common docker-selinux docker-engine 2. 安装依赖包&am…

Ubuntu安装SVN服务并结合内网穿透实现公网访问本地存储文件

&#x1f525;博客主页&#xff1a; 小羊失眠啦. &#x1f3a5;系列专栏&#xff1a;《C语言》 《数据结构》 《C》 《Linux》 《Cpolar》 ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&…

线性时间非比较类排序之桶排序

桶排序 桶排序也叫箱排序&#xff0c;1956年便开始使用&#xff0c;它可以算是计数排序的一个改进版本。 1. 算法思想 根据元素的特性将集合拆分为多个值域&#xff0c;我们称之为桶&#xff0c;将同一值域的元素存放在同一个桶内并进行桶内排序使其处于有序状态。如果每个桶…

华为 Huawei 交换机 黑洞MAC地址的作用和配置示例

黑洞mac作用&#xff1a;某交换机上配置某个PC的mac地址为黑洞mac&#xff0c;那么这台PC发出来的包都会被交换机丢弃&#xff0c;不会被转发到网络中。 组网需求&#xff1a; 如 图 2-13 所示&#xff0c;交换机 Switch 收到一个非法用户的访问&#xff0c;非法用户的 MAC 地址…

Docker容器监控-CIG

目录 一、CIG说明 1. CAdvisor 2. InfluxDB 3. Grafana 二、环境搭建 1. 创建目录 2. 编写 docker-compose.yml 3. 检查并运行容器 三、进行测试 1. 查看 influxdb 存储服务 是否能正常访问 2. 查看 cAdvisor 收集服务能否正常访问 3. 查看 grafana 展现服务&#…

Java毕业设计-基于springboot的网上书店管理系统-第75期

获取源码资料&#xff0c;请移步从戎源码网&#xff1a;从戎源码网_专业的计算机毕业设计网站 项目介绍 基于springboot的网上书店管理系统&#xff1a;前端thymeleaf、js、layui&#xff0c;后端 maven、springmvc、spring、mybatis&#xff0c;集成书籍管理、分类管理、订单…

Windows 安装 MySQL 最新最简教程

Windows 安装 MySQL 最新最简教程 官网地址 https://dev.mysql.com/downloads/mysql/下载 MySQL zip 文件 配置 MySQL1、解压文件 2、进入 bin 目录 搜索栏输入 cmd 回车进入命令行 C:\Users\zhong\Desktop\MySQL\mysql-8.3.0-winx64\mysql-8.3.0-winx64\bin 注意这里是你自己…