Mysql 事务管理问题.

3 篇文章 0 订阅

问题描述:

在做练习,配置spring拦截事务时,正确配置之后,发现事务始终不生效.

问题分析及查找原因:

1. spring的配置文件.

<!-- 开启事务注解驱动 -->
	<tx:annotation-driven />

	<!-- 事务管理器 -->
	<bean name="transactionManager"
		class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
		<property name="dataSource" ref="dataSource"></property>
	</bean>
	
	
	<!-- PROPAGATION_SUPPORTS: 如果已经存在事务,则加入事务;如果没有事务,则以非事务的方式执行; 
		 PROPAGATION_MANDATOR: 使用当前事务, 如果没有, 则抛出异常; 
		 PROPAGATION_REQUIRED: 支持当前事务,如果当前没有事务,就新建一个事务 
		 PROPAGATION_REQUIRED_NEW: 新建事务,如果当前有事务, 则挂起; 
		 ROPAGATION_NOT_SUPPORTED:以非事务的方式执行, 如果当前有事务, 则挂起; 
		 PROPAGATION_NEVER:以非事务的方式执行, 如果当前有事务,则抛出异常; 
		 +/-Exception:
		 + 表示异常出现时事物提交 - 表示异常出现时事务回滚 -->
	<tx:advice id="userTxAdvice" transaction-manager="transactionManager">
		<tx:attributes>
			<tx:method name="remove*" propagation="REQUIRED" read-only="false"
				rollback-for="java.lang.RuntimeException,java.lang.Exception" />
			<tx:method name="create*" propagation="REQUIRED" read-only="false"
				rollback-for="java.lang.RuntimeException,java.lang.Exception" />
			<tx:method name="save*" propagation="REQUIRED" read-only="false"
				rollback-for="java.lang.RuntimeException,java.lang.Exception" />	
			<tx:method name="update*" propagation="REQUIRED" read-only="false"
				rollback-for="java.lang.RuntimeException,java.lang.Exception" />
			<tx:method name="get*" propagation="SUPPORTS" read-only="true"/>
		</tx:attributes>
	</tx:advice>

	<aop:config>
		<aop:pointcut id="pc"
			expression="execution(public * com.snail.vipsys.services..*(..)) and this(com.snail.vipsys.common.aop.tx.Txable)" /> <!--把事务控制在Service层 -->
		<aop:advisor pointcut-ref="pc" advice-ref="userTxAdvice" />
	</aop:config>

这个配置没有问题的,因为之前已经配置过很多次了. 可是在测试时发生异常时数据始终是保存进去了.

又反反复复检查了配置文件,都没有发现哪里有配置错误,最后干脆将spring 事务管理的DEBUG级别的日志打印出来,结果在日志中看到"Switching JDBC Connection [org.apache.commons.dbcp.PoolableConnection@7968f74] to manual commit"这样一段英文,大概意思是已经被手动提交了,后面的日志显示spring的事务确实是回滚了的.

这样终于可以确定我的配置是没有问题的.

2.那问题可能在数据源的配置上了,通过查看数据源类(org.apache.commons.dbcp.BasicDataSource)的源码发现,其默认的配置是自动提交了.所以又在其配置中增加了配置其默认不自动提交.<property name="defaultAutoCommit" value="false" />    结果发现"Switching JDBC Connection [org.apache.commons.dbcp.PoolableConnection@7968f74] to manual commit"这条日志不再出现了,也正常回滚了.可是再查数据表时,发现数据还是正常插入到数据表中了.

3.于是问题又可能在数据库的设置上.我使用的数据库是MySQL5.5的,通过在网上查询发现,MySQL默认是自动提交的(show variables like 'autocommit'; --- 查看自动提交的配置,set autocommit = 0; -- 设置不自动提交).  于是将数据库的设置设置为不自动提交.再测试结果发现还是一样的正常插入到数据表了.

4.继续在网上查询MySQL的自动提交的相关资料,发现有人说使用非root账号就可以.所以又临时创建了一个测试账号,并设置成不自动提交. 再测试,依然还是添加到数据表中了.

5.继续查资料,最终发现了原来MySQL有引擎一说.

MySQL默认的存储引擎是MyISAM,MyISAM存储引擎不支持事务处理,所以改变autocommit没有什么作用。但不会报错,所以要使用事务处理的童鞋一定要确定你所操作的表示支持事务处理的,如InnoDB。如果不知道表的存储引擎可以通过查看建表语句查看建表的时候有没有指定事务类型的存储引擎,如果没有指定存储引擎默认则是MyISAM不支持事务的存储引擎。

原来MySQL为了效率高,默认使用的是不支持事务的MyISAm引擎, 必须改成InnoDB的引擎才支持使用.使用alter table table_name engine=innodb;  命令将引擎换了.再测试终于没有数据插入到数据表中了.


总结:

使用spring+MySQL配置事务管理时,需要做如下几步:

1.设置数据源默认不自动提交;

2.正确配置spring的事务管理;

           3.设置MySQL为不自动提交.

4.设置MySQL的引擎为innoDB



  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Node.js中使用MySQL服务时,可以通过事务管理来执行多条SQL操作,以确保原子性和数据一致性。事务是将一组操作视为一个单独的单元来执行,要么全部成功,要么全部失败。在Node.js中,可以使用Promise来封装事务操作。 首先,需要导入MySQL的连接池对象,例如使用`const pool = require("../db/mysql")`导入pool对象。然后,可以封装一个执行事务的函数`execTransaction`,该函数接受一个包含多个SQL语句的数组作为参数。 在`execTransaction`函数中,首先通过连接池获取一个数据库连接,然后开始事务。接下来,将所有需要执行的SQL语句封装为Promise数组,每个Promise代表一个SQL执行操作。在Promise中,使用连接对象的`query`方法执行SQL语句,并将结果返回。 使用`Promise.all`来等待所有的SQL操作完成,如果其中有任何一个操作出错,将回滚事务,并且释放连接。如果所有操作都成功执行,将提交事务,并释放连接。最后,通过`resolve`将结果返回。 这样,通过封装的`execTransaction`函数,可以在Node.js中方便地执行MySQL事务管理操作。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [Nodejs实现Mysql事务的解决方案](https://blog.csdn.net/weixin_41464806/article/details/106446449)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* [NodeJs使用Mysql模块实现事务处理实例](https://download.csdn.net/download/weixin_38678796/14858646)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

蜗牛_snail

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值