优化@Transactional事务性能(LazyConnectionDataSourceProxy)

news/2025/3/9 10:52:04/文章来源:https://www.cnblogs.com/tiancai/p/18680715

背景

在项目开发中,使用 @Transactional 注解来管理事务非常方便,且优雅。但是也存在一个问题:长事务问题

很多被 @Transactional 标记的方法,实际上并不需要进行数据库操作,或者说,它们在执行的很长一段时间内都不会真正触发数据库访问。

举个例子,我们的业务逻辑可能如下:

@Service
public class OrderService {@Transactionalpublic void processOrder(Long orderId) {// 1. 复杂的业务校验(纯计算,无数据库操作)validateOrder(orderId);// 2. 调用外部服务(可能是一个耗时操作,比如支付、物流)callExternalService(orderId);// 3. 数据库操作(此时才真正需要数据库连接)updateOrderStatus(orderId);}
}

在这个 processOrder() 方法中:

  • 只有 updateOrderStatus(orderId) 这一行代码涉及数据库操作。
  • 但由于 @Transactional 的作用,Spring 在 方法一开始 就会从数据库连接池获取连接,即便前面两步根本不需要用到数据库。
  • 如果 validateOrder()callExternalService() 是一个耗时操作,就会导致数据库连接长时间被占用,影响系统吞吐量。

这种情况下,我们可以使用 LazyConnectionDataSourceProxy 来优化数据库连接的获取时机,避免不必要的连接占用。


解决办法:使用Spring提供的LazyConnectionDataSourceProxy

Spring给我们提供了一个非常好的利器:可以解决此类长事务的问题,来看具体用法示例, 后面我会分析它的原理

LazyConnectionDataSourceProxy 用法示例

在 Spring 配置中,我们可以用 LazyConnectionDataSourceProxy 代理现有的 DataSource,使得数据库连接不会在事务开始时立即获取,而是在真正执行 SQL 时才获取。

Spring Boot 配置示例
@Configuration
public class DataSourceConfig {@Beanpublic DataSource realDataSource() {HikariDataSource dataSource = new HikariDataSource();dataSource.setJdbcUrl("jdbc:mysql://localhost:3306/test");dataSource.setUsername("root");dataSource.setPassword("password");return dataSource;}@Beanpublic DataSource dataSource() {return new LazyConnectionDataSourceProxy(realDataSource());}@Beanpublic PlatformTransactionManager transactionManager() {return new DataSourceTransactionManager(dataSource());}
}

在这里,我们用 LazyConnectionDataSourceProxy 包装了真正的数据库 DataSource,然后让 TransactionManager 使用这个代理数据源。


LazyConnectionDataSourceProxy 的原理

LazyConnectionDataSourceProxy 的核心机制就是 延迟获取数据库连接,它的主要逻辑如下:

  1. 事务开启时,Spring 事务管理器不会立即从连接池获取数据库连接,而是返回一个 Connection 代理对象。
  2. 当代码真正执行数据库操作(如 prepareStatement()createStatement())时,代理对象才会去获取真实的数据库连接。
  3. 事务提交或回滚时,如果 Connection 代理对象从未真正执行过 SQL,则根本不会创建数据库连接,从而节省资源。

关键源码如下:

    public Connection getConnection() throws SQLException {// 返回的是代理对象return (Connection)Proxy.newProxyInstance(ConnectionProxy.class.getClassLoader(), new Class[]{ConnectionProxy.class}, new LazyConnectionInvocationHandler());}public Connection getConnection(String username, String password) throws SQLException {return (Connection)Proxy.newProxyInstance(ConnectionProxy.class.getClassLoader(), new Class[]{ConnectionProxy.class}, new LazyConnectionInvocationHandler(username, password));}

其中,getConnection() 方法返回的 Connection 是一个 JDK 动态代理对象,它会拦截 prepareStatement()createStatement() 等方法,直到真正执行 SQL 时才去获取连接。这里我们主要关注LazyConnectionInvocationHandler类的invoke方法,因为根据JDK动态代理的特性, 当Connection代理对象的方法执行时,会触发到invoke方法

LazyConnectionInvocationHandler源码解析

主要需要关注LazyConnectionInvocationHandlerinvoke方法,关键代码部分有增加注释,感兴趣的可以看看

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {// 以下的这些 if …… else 的判断,主要就是通过排除法锁定prepareStatement 方法// 只要 method.getName 不为:prepareStatement 则spring都使用了硬编码做了模拟方法实现,可以仔细分析一下源代码if (method.getName().equals("equals")) {return proxy == args[0];} else if (method.getName().equals("hashCode")) {return System.identityHashCode(proxy);} else {if (method.getName().equals("unwrap")) {if (((Class)args[0]).isInstance(proxy)) {return proxy;}} else if (method.getName().equals("isWrapperFor")) {if (((Class)args[0]).isInstance(proxy)) {return true;}} else if (method.getName().equals("getTargetConnection")) {return this.getTargetConnection(method);}
// 当没有执行 prepareStatement 方法,则 hasTargetConnection() 的返回值恒为 false // 但是该 if 逻辑内部恰好排除了 prepareStatement 方法的执行,// 也就是说当 Connection 执行 prepareStatement 时会进入else 的处理逻辑if (!this.hasTargetConnection()) {if (method.getName().equals("toString")) {return "Lazy Connection proxy for target DataSource [" + LazyConnectionDataSourceProxy.this.getTargetDataSource() + "]";}if (method.getName().equals("getAutoCommit")) {if (this.autoCommit != null) {return this.autoCommit;}} else {if (method.getName().equals("setAutoCommit")) {this.autoCommit = (Boolean)args[0];return null;}if (method.getName().equals("getTransactionIsolation")) {if (this.transactionIsolation != null) {return this.transactionIsolation;}} else {if (method.getName().equals("setTransactionIsolation")) {this.transactionIsolation = (Integer)args[0];return null;}if (method.getName().equals("isReadOnly")) {return this.readOnly;}if (method.getName().equals("setReadOnly")) {this.readOnly = (Boolean)args[0];return null;}if (method.getName().equals("getHoldability")) {return this.holdability;}if (method.getName().equals("setHoldability")) {this.holdability = (Integer)args[0];return null;}if (method.getName().equals("commit")) {return null;}if (method.getName().equals("rollback")) {return null;}if (method.getName().equals("getWarnings")) {return null;}if (method.getName().equals("clearWarnings")) {return null;}if (method.getName().equals("close")) {this.closed = true;return null;}if (method.getName().equals("isClosed")) {return this.closed;}if (this.closed) {throw new SQLException("Illegal operation: connection is closed");}}}}
// 这里才真正获取数据库连接try {return method.invoke(this.getTargetConnection(method), args);} catch (InvocationTargetException var5) {throw var5.getTargetException();}}}

原理一句话总结:

使用JDK动态代理connection, 使得原有@Transaction注解在方法第一行就会获取数据库连接的模式,改变为只有当真正需要执行sql时,才获取连接,也即懒加载模式获取连接,这样就大大减少了连接的占用时间,也可以缩短事务的占用时间


性能优化效果

在引入 LazyConnectionDataSourceProxy 之后,数据库连接的获取时机由 “事务开始” 变为 首次执行 SQL 时,这带来了几个优化点:

  1. 减少数据库连接池的压力:原本事务开启就会立即获取数据库连接,现在只有在执行 SQL 语句时才会获取,避免连接池资源被长期占用。
  2. 提高系统吞吐量:在高并发场景下,多个 @Transactional 方法可能会并发执行,如果它们的前半部分是耗时操作,LazyConnectionDataSourceProxy 可以减少数据库连接的浪费,从而提高整体性能。
  3. 优化 @Transactional 在无数据库操作事务中的表现:如果 @Transactional 方法最终根本没有执行 SQL,那么 LazyConnectionDataSourceProxy 会使得这个事务 从头到尾都不会获取数据库连接,减少了不必要的资源开销。

实际使用中的注意点

  1. LazyConnectionDataSourceProxy 仅仅优化了数据库连接的获取时机,并不会改变事务的传播机制。如果方法确实需要事务,@Transactional 仍然是必不可少的。
  2. 不能直接替代数据库连接池LazyConnectionDataSourceProxy 需要和 HikariCP、Druid 等连接池配合使用,保证高效的连接管理。
  3. 不适用于直接管理 Connection 的代码。如果代码中有 connection.unwrap()connection.getMetaData() 这样的调用,可能会导致代理对象提前获取连接,影响延迟获取的效果。

总结

在 Spring 事务管理中,@Transactional 非常方便,但如果方法内部的 非数据库操作占比高,就可能造成数据库连接的长期占用,事务的长时间占用,影响系统性能

使用 LazyConnectionDataSourceProxy 可以 延迟数据库连接的获取,让事务管理更加高效,减少数据库连接池的压力,特别适用于高并发、复杂业务逻辑的场景。

对于 @Transactional 方法中包含大量 非数据库操作(如远程调用、复杂计算) 的情况,这个优化手段值得尝试!

转自:https://blog.csdn.net/qq_35249342/article/details/145245085

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ElasticSearch Java 使用

目录创建工程,导入坐标创建索引 index创建映射 mapping建立文档 document建立文档(通过 XContentBuilder)建立文档(使用 Jackson 转换实体)1)添加jackson坐标2)创建 Article 实体3)代码实现查询文档操作关键词查询字符串查询使用文档 ID 查询文档查询文档分页操作批量插…

IntelliJ IDEA 2024.3 Java开发工具

IntelliJ IDEA 2024.3 Java开发工具 JetBrains IntelliJ IDEA 2024 mac,是一款Java开发工具,IntelliJ IDEA 凭借无与伦比的 Java 和 Kotlin 支持脱颖而出。从一开始就支持尖IDEA 2024.3 中文版开发工具端语言功能,保持领先地位。IntelliJ IDEA 对您的代码了如指掌,利用这些…

AI应用实战课学习总结(6)分类算法分析实战

本文介绍了机器学习中的分类场景问题,常用的分类算法 以及 分类和回归的简单对比,最后通过一个医疗数据诊断分类的案例做了一次实战,相信对你理解分类应用应该有所帮助。大家好,我是Edison。 最近入坑黄佳老师的《AI应用实战课》,记录下我的学习之旅,也算是总结回顾。 今…

深入浅出索引(上)

1.什么是数据库索引,索引是干什么用的? 对于数据库的表而言,索引其实就是它的“目录”。 2.索引的三种实现方式?(暂时介绍3种) ①哈希表索引:哈希表是一种以键 - 值(key-value)存储数据的结构,我们只要输入待查找的值即 key,就可以找到其对应的值即 Value。哈希的思…

【推荐】一款开源且功能丰富、技术先进的在线项目任务管理系统

项目介绍 DooTask是一款基于PHP + Vue开源的、功能丰富、技术先进、安全性高的开源在线项目任务管理系统,适合各种规模的团队使用。提供各类文档协作工具、在线思维导图、在线流程图、项目管理、任务分发、即时IM,文件管理等工具;同时消息功能使用非对称加密技术让你的沟通更…

《SpringBoot》自动装配原理(简单易懂)

引入 先看SpringBoot的主配置类 @SpringBootApplication public class DemoApplication{public static void main(String[] args){SpringApplication.run(StartEurekaApplication.class, args);} }@SpringBootApplication 点进@SpringBootApplication来看,发现@SpringBootAppl…

VMware Data Services Manager 2.2 发布 - 数据库管理和数据服务管理

VMware Data Services Manager 2.2 发布 - 数据库管理和数据服务管理VMware Data Services Manager 2.2 发布 - 数据库管理和数据服务管理 在 vSphere 环境中按需配置和自动管理 PostgreSQL 和 MySQL 数据库 请访问原文链接:https://sysin.org/blog/vmware-data-services-mana…

PyMuPDF4LLM : 提取 PDF 数据的终结者

你知道处理 PDF 文件时会遇到什么,尤其是那些包含复杂研究内容或者有表格、图像和额外信息的文档。对于 AI 领域的人,特别是那些在调整大型语言模型(LLM)或构建知识检索系统(比如 RAG)的人来说,提取正确的数据非常重要。然而,老实说,并不是所有的 PDF 提取工具都能做到…

【Java开发】LLM实战:使用LangChain4j构建本地RAG系统

一、引言二、基本概念2.1 什么是RAG2.2 LangChain4j简介2.3 大模型开发 vs. 传统JAVA开发三、实战经验3.1 环境搭建3.1.1 向量库(Chroma)  3.1.2 集成LangChain4j3.2 程序编写3.2.1 项目结构3.2.2 知识采集3.2.3 文档切分  3.2.4 文本向量化3.2.5 向量…

3D NAND闪存技术的架构和工艺集成概述(下)

另一种具有FG的3D NAND结构是HC-FG结构,如图8a所示。与DC-SF结构不同,水平信道堆叠在HC-FG架构中;然而,单元格没有被CG包围。因此,FG单元可以通过沟道优先工艺堆叠,类似于传统的2D平面灰阵列,并且可以以低成本实现3D结构。此外,如图8b所示,HC-FG结构与层选择晶体管(L…

3D NAND闪存技术的架构和工艺集成概述(上)

Architecture and Process Integration Overview of 3D NAND Flash Technologies 3D NAND闪存技术的架构和工艺集成概述 摘要:在过去的几十年里,NAND闪存一直是最成功的非易失性存储技术之一,由于其高可扩展性和可靠的开关特性,它在电子设备中得到了广泛的应用。为了克服平…

推荐4本书《AI芯片开发核心技术详解》、《智能汽车传感器:原理设计应用》、《TVM编译器原理与实践》、《LLVM编译器原理与实践》,谢谢

4本书推荐《AI芯片开发核心技术详解》、《智能汽车传感器:原理设计应用》、《TVM编译器原理与实践》、《LLVM编译器原理与实践》由清华大学出版社资深编辑赵佳霓老师策划编辑的新书《AI芯片开发核心技术详解》已经出版,京东、淘宝天猫、当当等网上,相应陆陆续续可以购买。该…