spring事务在readonly上的一个疏忽还是一个bug

2024-06-15 08:18

本文主要是介绍spring事务在readonly上的一个疏忽还是一个bug,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

问题描述:
http://www.iteye.com/topic/319768

经过测试加上源代码,发现, 如果你使用的是DataSourceTransactionManager来管理事务,readonly是不起作用的:(HibernateTransactionManager是起作用的,后面有源码分析)
Xml代码   收藏代码
  1. <bean id="transactionManager"  
  2.     class="org.springframework.jdbc.datasource.DataSourceTransactionManager">  
  3.     <property name="dataSource" ref="dataSource" />  
  4. </bean>  
  5.   
  6. <tx:advice id="txAdvice" transaction-manager="transactionManager">  
  7.     <tx:attributes>  
  8.         <tx:method name="search*" read-only="true" />  
  9.     </tx:attributes>  
  10. </tx:advice>  


先看源码:
1. DataSourceTransactionManager.doBegain 方法:
Java代码   收藏代码
  1. protected void doBegin(Object transaction, TransactionDefinition definition) {  
  2.         DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;  
  3.         Connection con = null;  
  4.   
  5.         try {  
  6.             if (txObject.getConnectionHolder() == null ||  
  7.                     txObject.getConnectionHolder().isSynchronizedWithTransaction()) {  
  8.                 Connection newCon = this.dataSource.getConnection();  
  9.                 if (logger.isDebugEnabled()) {  
  10.                     logger.debug("Acquired Connection [" + newCon + "] for JDBC transaction");  
  11.                 }  
  12.                 txObject.setConnectionHolder(new ConnectionHolder(newCon), true);  
  13.             }  
  14.   
  15.             txObject.getConnectionHolder().setSynchronizedWithTransaction(true);  
  16.             con = txObject.getConnectionHolder().getConnection();  
  17.   
  18.             Integer previousIsolationLevel = DataSourceUtils.prepareConnectionForTransaction(con, definition);  
  19.             txObject.setPreviousIsolationLevel(previousIsolationLevel);  
  20.   
  21.             // Switch to manual commit if necessary. This is very expensive in some JDBC drivers,  
  22.             // so we don't want to do it unnecessarily (for example if we've explicitly  
  23.             // configured the connection pool to set it already).  
  24.             if (con.getAutoCommit()) {  
  25.                 txObject.setMustRestoreAutoCommit(true);  
  26.                 if (logger.isDebugEnabled()) {  
  27.                     logger.debug("Switching JDBC Connection [" + con + "] to manual commit");  
  28.                 }  
  29.                 con.setAutoCommit(false);  
  30.             }  
  31.             txObject.getConnectionHolder().setTransactionActive(true);  
  32.               
  33.             int timeout = determineTimeout(definition);  
  34.             if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {  
  35.                 txObject.getConnectionHolder().setTimeoutInSeconds(timeout);  
  36.             }  
  37.   
  38.             // Bind the session holder to the thread.  
  39.             if (txObject.isNewConnectionHolder()) {  
  40.                 TransactionSynchronizationManager.bindResource(getDataSource(), txObject.getConnectionHolder());  
  41.             }  
  42.         }  
  43.   
  44.         catch (SQLException ex) {  
  45.             DataSourceUtils.releaseConnection(con, this.dataSource);  
  46.             throw new CannotCreateTransactionException("Could not open JDBC Connection for transaction", ex);  
  47.         }  
  48.     }  



2. HibernateTransactionManager.doBegain方法中的一段:

Java代码   收藏代码
  1. protected void doBegin(Object transaction, TransactionDefinition definition)  
  2.     {  
  3. //省略一些代码  
  4.  if(!definition.isReadOnly() && !txObject.isNewSession())  
  5.         {  
  6.             FlushMode flushMode = session.getFlushMode();  
  7.             if(flushMode.lessThan(FlushMode.COMMIT))  
  8.             {  
  9.                 session.setFlushMode(FlushMode.AUTO);  
  10.                 txObject.getSessionHolder().setPreviousFlushMode(flushMode);  
  11.             }  
  12.         }  
  13. }  


可以看出, DataSourceTransactionManager.doBegain方法中没有一处来判断readonly的,所以不起作用,如果我加上:
Java代码   收藏代码
  1. protected void doBegin(Object transaction, TransactionDefinition definition) {  
  2. //硬代码:con.setReadOnly(true)  
  3. con.setReadOnly(true);  
  4.   
  5. int timeout = determineTimeout(definition);  
  6.             if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {  
  7.                 txObject.getConnectionHolder().setTimeoutInSeconds(timeout);  
  8.             }  
  9. }  


那么可以成功,看后台:
Xml代码   收藏代码
  1. nested exception is java.sql.SQLException: ORA-01456: may not perform insert/delete/update operation inside a READ ONLY transaction  
  2.   
  3.     at org.springframework.jdbc.support.SQLStateSQLExceptionTranslator.translate(SQLStateSQLExceptionTranslator.java:124)  


结论:如果你使用DataSourceTransactionManager来控制事务,即使你开启了readonly也是无效的。当然你可以自己修改源代码,加上readonly的判断

但为何spring不控制在DataSourceTransactionManager的readonly,而开启在HibernateTransactionManager的呢,这是个问题!

这篇关于spring事务在readonly上的一个疏忽还是一个bug的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1062911

相关文章

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

Redis事务与数据持久化方式

《Redis事务与数据持久化方式》该文档主要介绍了Redis事务和持久化机制,事务通过将多个命令打包执行,而持久化则通过快照(RDB)和追加式文件(AOF)两种方式将内存数据保存到磁盘,以防止数据丢失... 目录一、Redis 事务1.1 事务本质1.2 数据库事务与redis事务1.2.1 数据库事务1.

SpringCloud集成AlloyDB的示例代码

《SpringCloud集成AlloyDB的示例代码》AlloyDB是GoogleCloud提供的一种高度可扩展、强性能的关系型数据库服务,它兼容PostgreSQL,并提供了更快的查询性能... 目录1.AlloyDBjavascript是什么?AlloyDB 的工作原理2.搭建测试环境3.代码工程1.

Java调用Python代码的几种方法小结

《Java调用Python代码的几种方法小结》Python语言有丰富的系统管理、数据处理、统计类软件包,因此从java应用中调用Python代码的需求很常见、实用,本文介绍几种方法从java调用Pyt... 目录引言Java core使用ProcessBuilder使用Java脚本引擎总结引言python

SpringBoot操作spark处理hdfs文件的操作方法

《SpringBoot操作spark处理hdfs文件的操作方法》本文介绍了如何使用SpringBoot操作Spark处理HDFS文件,包括导入依赖、配置Spark信息、编写Controller和Ser... 目录SpringBoot操作spark处理hdfs文件1、导入依赖2、配置spark信息3、cont

springboot整合 xxl-job及使用步骤

《springboot整合xxl-job及使用步骤》XXL-JOB是一个分布式任务调度平台,用于解决分布式系统中的任务调度和管理问题,文章详细介绍了XXL-JOB的架构,包括调度中心、执行器和Web... 目录一、xxl-job是什么二、使用步骤1. 下载并运行管理端代码2. 访问管理页面,确认是否启动成功

Java中的密码加密方式

《Java中的密码加密方式》文章介绍了Java中使用MD5算法对密码进行加密的方法,以及如何通过加盐和多重加密来提高密码的安全性,MD5是一种不可逆的哈希算法,适合用于存储密码,因为其输出的摘要长度固... 目录Java的密码加密方式密码加密一般的应用方式是总结Java的密码加密方式密码加密【这里采用的

Java中ArrayList的8种浅拷贝方式示例代码

《Java中ArrayList的8种浅拷贝方式示例代码》:本文主要介绍Java中ArrayList的8种浅拷贝方式的相关资料,讲解了Java中ArrayList的浅拷贝概念,并详细分享了八种实现浅... 目录引言什么是浅拷贝?ArrayList 浅拷贝的重要性方法一:使用构造函数方法二:使用 addAll(