Spring中的事务

一 事务
事务是一组操作的执行单元,针对数据库操作,事务管理的是一组SQL指令,事务内的操作要不全部成功,要不全部失败。比如执行过程中,如果有一条SQL语句没有执行成功,那么这一组操作都将全部回滚
事务特性(ACID):
Atomic(原子性):要么都成功,要么都失败
Consistent(一致性):数据不应该被破坏
Isolate(隔离性):用户间操作不相混淆
Durable(持久性):永久保存

1 编程式事务
编程式的事务,可以实现细粒度的事务控制,比如可以控制事务何时开始,何时结束等,但是spring中一般不提倡使用。Spring提供两种方式的编程式事务管理,分别是使用TransactionTemplate和直接使用PlatformTransactionManager。

2 声明式事务
在Spring中使用AOP思想,对目标方法动态织入事务处理操作,根据目标方法执行情况,提交或者回滚。在Spring中,只需要在配置文件中进行一系列配置(或者通过@Transactional注解),即可将相关方法纳入到事务管理中,这种做法降低了代码的耦合度,灵活方便。

Try{
数据库的操作
Comit()
}catch(Exception e){
Rollback()
}finally{
Close()
}

Spring使用声明式事务,结合AOP一起使用
Spring中的事务针对方法,一般将事务用在service层

二 事务配置
1 引入jar包
需要导入AOP相关jar文件,
还需要导入spring-tx-5.1.5.RELEASE.jar

2 引入名称空间

3 xml方式事务的配置

<!-- 1配置事务管理类 -->
<bean id="txManage" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
	<property name="dataSource" ref="dataSource"></property>
</bean>

<tx:advice id=“txAdvice” transaction-manager=“txManage”>
tx:attributes

<tx:method name=“get*” read-only=“true” propagation=“SUPPORTS”/>
<tx:method name=“add*” read-only=“false” propagation=“REQUIRED”/>
<tx:method name="*" propagation=“NOT_SUPPORTED”/>

	</tx:attributes>

</tx:advice>

aop:config

<aop:pointcut expression=“execution(* com.rr.xml.PersonService.*(…))” id=“pc”/>

	<!-- 通知 -->
	<aop:advisor advice-ref="txAdvice" pointcut-ref="pc"/>

</aop:config>

4 注解方式
@Transactional 既可以修饰方法,也可以修饰类

//rollbackFor=Exception.class
//rollbackFor 表示哪些异常回滚
//noRollbackFor 表示某些异常下不回滚
//声明事务的注解
@Transactional(readOnly=false,//是否只读,如果只有查询,设置为true,效率高
propagation=Propagation.REQUIRED, //事务的传播特性
isolation=Isolation.DEFAULT, //事务的隔离特性
timeout=3,//事务的超时时间,单位秒,默认-1
noRollbackFor=ArithmeticException.class //回滚规则

		)
public void addPerson(String name){}
<bean id="txManage" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
	<property name="dataSource" ref="dataSource"></property>
</bean>

<!-- 2 事务的注解扫描 -->
<tx:annotation-driven transaction-manager="txManage"/>

事务的传播特性:多个事务方法相互调用时,事务如何在这些方法间传播
传播特性 意义
REQUIRED 常用默认配置。业务方法运行时会开启事务。如果方法运行时,已经处在一个事务中,进行事务的合并
NOT_SUPPORTED 声明方法不需要事务。如果方法没有关联到一个事务,容器不会为它开启事务。如果方法在一个事务中被调用,该事务会被挂起,在方法调用结束后,原先的事务便会恢复执行
REQUIRES_NEW 不管是否存在事务,业务方法总会开启一个新的事务。如果方法已经运行在一个事务中,则原有事务会被挂起,新的事务会被创建,直到方法执行结束,新事务才算结束,原先的事务才会恢复执行
MANDATORY 业务方法只能在一个已经存在的事务中执行。如果业务方法在没有事务的环境下调用,容器就会抛出例外。
SUPPORTS 如果业务方法在某个事务范围内被调用,则方法成为该事务的一部分。如果业务方法在事务范围外被调用,则方法在没有事务的环境下执行
Never 指定业务方法绝对不能在事务范围内执行。如果业务方法在某个事务中执行,容器会抛出异常,只有业务方法没有关联到任何事务,才能正常执行
NESTED 如果一个活动的事务存在,则运行在一个嵌套的事务中. 如果没有活动事务, 则按REQUIRED属性执行.它使用了一个单独的事务, 这个事务拥有多个可以回滚的保存点。内部事务的回滚不会对外部事务造成影响。它只对DataSourceTransactionManager事务管理器起效

事务的隔离级别:用来限定事务内外的哪些改变是可见的,哪些是不可见的。低级别的隔离级一般支持更高的并发处理,并拥有更低的系统开销。
隔离级别 含义
DEFAULT 使用后端数据库默认的隔离级别(spring中的的选择项)
READ_UNCOMMITED 允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读
READ_COMMITTED 允许在并发事务已经提交后读取。可防止脏读,但幻读和 不可重复读仍可发生
REPEATABLE_READ 对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生。
SERIALIZABLE 完全服从ACID的隔离级别,确保不发生脏、幻、不可重复读。这在所有的隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的。

什么是脏数据,脏读,不可重复读,幻觉读?
1)脏读: 指当一个事务正在访问数据,并且对数据进行了修改,而这种修改还没有提交到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据。因为这个数据是还没有提交的数据, 那么另外一个事务读到的这个数据是脏数据,依据脏数据所做的操作可能是不正确的。
2)不可重复读: 指在一个事务内,多次读同一数据。在这个事务还没有结束时,另外一个事务也访问该同一数据。那么,在第一个事务中的两次读数据之间,由于第二个事务的修改,那么第一个事务两次读到的数据可能是不一样的。这样就发生了在一个事务内两次读到的数据是不一样的,因此称为是不可重复读。
3)幻觉读: 指当事务不是独立执行时发生的一种现象,例如第一个事务对一个表中的数据进行了修改,这种修改涉及到表中的全部数据行。同时,第二个事务也修改这个表中的数据,这种修改是向表中插入一行新数据。那么,以后就会发生操作第一个事务的用户发现表中还有没有修改的数据行,就好象发生了幻觉一样。

mysql默认的事务处理级别是’REPEATABLE-READ’,也就是可重复读
查看当前会话隔离级别
select @@tx_isolation;

查看系统当前隔离级别
select @@global.tx_isolation;

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值