了解 Spring Data JPA 事务的配置以及 @Transactional 注解的使用方法.Spring 使用注解方式进行事务管理
Spring Data JPA 是 Spring 基于 ORM 框架、JPA 规范的基础上封装的一套 JPA 应用框架,可使开发者用极简的代码即可实现对数据的访问和操作。它提供了包括增删改查等在内的常用功能,且易于扩展。以下介绍JPA事务
1.事务基础:
事务的基本概念:即访问并可能更新数据库中各种数据项的一个程序执行单元。
需要管理事务的执行,要么成功、要么失败,一但失败,所有操作将回滚到初始状态,一但成功,则进行持久化。
事务特性ACID:原子性、一致性、隔离性(并发执行的事务不能相互干扰)、持久性(对数据库的改变是永久性)
Spring Data JPA事务分为:JTA事务(分布式事务,多种数据库)、RESOURCE_LOCAL事务(本地事务,数据库级别,仅支持一种数据库)
2.数据库事务并发带来的问题
脏读:一句话表达事务读取了其他并发事务未提交的数据
事务B读取了事务A未提交的数据,事务B按未提交的数据进行执行并提交。而事务A又对数据进行修改后再提交。这样事务B读取的数据与事务A提交的数据不一致。
不可重复读:同一个事务先后两次或两次以上读取同一数据,结果不一样。
事务C读取了数据,事务还没有提交。
事务D修改了数据,并提交事务。随后,事务C再次读取时,发现数据变了。
幻读:跟不可重复读类似,侧重记录的数量(行数),不可重复读侧重于数据的值。
3.事务的传播特性:
FunctionA调用FunctionB
1. PROPAGATION_REQUIRED: 如果存在一个事务,则支持当前事务。如果没有事务则开启
2. PROPAGATION_SUPPORTS: 如果存在一个事务,支持当前事务。如果没有事务,则非事务的执行
3. PROPAGATION_MANDATORY: 如果已经存在一个事务,支持当前事务。如果没有一个活动的事务,则抛出异常。
4. PROPAGATION_REQUIRES_NEW: 总是开启一个新的事务。如果一个事务已经存在,则将这个存在的事务挂起。
5. PROPAGATION_NOT_SUPPORTED: 总是非事务地执行,并挂起任何存在的事务。
6. PROPAGATION_NEVER: 总是非事务地执行,如果存在一个活动事务,则抛出异常
7. PROPAGATION_NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中. 如果没有活动事务, 则按TransactionDefinition.PROPAGATION_REQUIRED 属性执行
4.事务的隔离级别:
1. ISOLATION_DEFAULT: 这是一个PlatfromTransactionManager默认的隔离级别,使用数据库默认的事务隔离级别.另外四个与JDBC的隔离级别相对应
2. ISOLATION_READ_UNCOMMITTED: 这是事务最低的隔离级别,它充许令外一个事务可以看到这个事务未提交的数据。这种隔离级别会产生脏读,不可重复读和幻像读。
3. ISOLATION_READ_COMMITTED: 保证一个事务修改的数据提交后才能被另外一个事务读取。另外一个事务不能读取该事务未提交的数据
4. ISOLATION_REPEATABLE_READ: 这种事务隔离级别可以防止脏读,不可重复读。但是可能出现幻像读。它除了保证一个事务不能读取另一个事务未提交的数据外,还保证了避免下面的情况产生(不可重复读)。
5. ISOLATION_SERIALIZABLE 这是花费最高代价但是最可靠的事务隔离级别。事务被处理为顺序执行。除了防止脏读,不可重复读外,还避免了幻像读。
通常使用数据库的默认隔离级别即可,基本不需要进行设置
MYSQL: 默认为REPEATABLE_READ级别
SQLSERVER: 默认为READ_COMMITTED
5.配置事务
先在spring配置文件中引入<tx:>命名空间
5.1通过注解配置事务:
<!-- 事务管理器配置, Hibernate单数据源事务 -->
<bean id="defaultTransactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
<property name="sessionFactory" ref="sessionFactory" />
</bean>
<!-- 使用annotation定义事务 -->
<tx:annotation-driven transaction-manager="defaultTransactionManager" proxy-target-class="true" />
配置完之后,就可通过@Transactional 注解的bean自动配置为声明式事务支持 .
注意以下问题:
a.<tx:annotation-driven/>元素的出现开启了事务行为,@Transactional 是元数据标记,只有在前者配置后,再使用@Transactional才能开启事务
b.@Transactional 可以添加到Dao层或Servive层(具体的实现类)的 public 可见度的方法上,而不要使用在类所要实现的任何接口上。你当然可以在接口上使用 @Transactional 注解,但是这将只能当你设置了基于接口的代理时它才生效。因为注解是 不能继承 的
5.2通过AOP配置事务:
<!--事务的 AOP 配置-->
<!--advisor-->
<tx:advice id="applicationAdvisor" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="*" propagation="REQUIRED"/>
<tx:method name="load*" propagation="REQUIRED" read-only="false" isolation="DEFAULT" rollback-for="UserNotFoundException,UserNotFound2Exception"/>
<tx:method name="is*" propagation="REQUIRED" read-only="false"/>
</tx:attributes>
</tx:advice>
<aop:config>
<aop:advisor advice-ref="applicationAdvisor" pointcut="execution(* com.mm.*.service.impl.*.*(..))"/>
</aop:config>