Spring Data JPA 是 Spring 基于 ORM 框架、JPA 规范的基础上封装的一套 JPA 应用框架,可以使开发者用极简的代码便可实现对数据的访问和操做。它提供了包括增删改查等在内的经常使用功能,且易于扩展。如下介绍JPA事务spring
事务的基本概念:即访问并可能更新数据库中各类数据项的一个程序执行单元。数据库
须要管理事务的执行,要么成功、要么失败,一但失败,全部操做将回滚到初始状态,一但成功,则进行持久化。session
事务特性ACID:原子性、一致性、隔离性(并发执行的事务不能相互干扰)、持久性(对数据库的改变是永久性)并发
Spring Data JPA事务分为:JTA事务(分布式事务,多种数据库)、RESOURCE_LOCAL事务(本地事务,数据库级别,仅支持一种数据库)app
脏读:一句话表达事务读取了其余并发事务未提交的数据框架
事务B读取了事务A未提交的数据,事务B按未提交的数据进行执行并提交。而事务A又对数据进行修改后再提交。这样事务B读取的数据与事务A提交的数据不一致。分布式
不可重复读:同一个事务前后两次或两次以上读取同一数据,结果不同。spa
事务C读取了数据,事务尚未提交。hibernate
事务D修改了数据,并提交事务。随后,事务C再次读取时,发现数据变了。3d
幻读:跟不可重复读相似,侧重记录的数量(行数),不可重复读侧重于数据的值。
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 属性执行
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
先在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>