Mysql 事务管理问题.

问题描述:

在做练习,配置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



猜你喜欢

转载自blog.csdn.net/shui878412/article/details/51714008