事务管理——原子性、一致性、隔离性、持久性

理解spring对事务管理的支持

Spring提供对编码式和声明式事务管理的支持。编码式事务允许用户在代码中精确定义事务的边界,而声明式事务(基于AOP,面向切面的特性)有助于用户将操作与事务规则进行解耦。

Spring 通过回调机制将实际的事务实现从事务性的代码中抽象出来。实际上,spring对事务的支持甚至不需要JTA的实现。若应用程序只使用一种持久化资源,spring可以使用持久化机制所提供的事务性支持,包括JDBC,Hibernate以及Java持久化API.。但是如果应用程序的事务跨多个资源,那么spring会使用第三方的JTA实现来支持分布式事务。

JDBC事务

<bean  id=”transactionManager”  class=”org.springframework.jdbc.datasource.DataSourceTransactionManager”>

<property name=”dataSource”  ref=”dataSource”/>

</bean>

在幕后,DataSourceTransactionManager通过java.sql.connection来管理事务,而后者是通过DataSource获取到的。通过调用链接的commit()方法来提交事务。事务失败时通过调用rollback()方法进行回滚。

Hibernate 事务

<bean  id=”transactionManager”  class=”org.springframework.orm.hibernate3.HibernateTransactionManager”>

<property name=”sessionFactory”  ref=”sessionFactory” />

</bean>

HibernateTransactionManager将事务管理委托给org.hibernate.Transaction对象,后者从Hibernate Session中获取到。当事务成功时,HibernateTransactionManager会调用Transaction的commit()方法。反之,调用rollback()方法。

Java持久化API事务(JPA -Java Persistence API)

<bean id=”transactionManager” class=”org.springframework.orm.jpa.JpaTransactionManager”>

<property name=”entityManagerFactory” ref=”entityManagerFactory” />

</bean>

JpaTransactionManager需要装配一个JPA实体管理工厂(javax.persistence.EntityManagerFactory接口的任意实现)。JpaTransactionManager将与由工厂所产生的JPA EntityManager合作来构建事务。

除了将事务应用于JPA操作,JpaTransactionManager还支持将事务应用于简单的JDBC操作之中,这些JDBC操作所使用的DataSource与entityManagerFactory所使用的DataSource必须是相同的。为了做到这一点,JpaTransactionManager必须装配一个JpaDialect的实现。

若已经配置了EclipseLinkJpaDialect

<bean id=”jpaDialect” class=”org.springframework.ora.jpa.vendor.EclipseLinkJpaDialect” />

则将jpaDialect装配到JpaTransactionManager中,即

<bean id=”transactionManager”class=org.springframework.orm.jpa.JpaTransactionManager”>

<property name=”entityManagerFactory”  ref=”entityManagerFactory” />

<property name=”jpaDialect”  ref=”jpaDialect” />

</bean>

但是jpaDialect的实现必须同时支持JPA/JDBC访问,而DefaultJpaDialect不支持。

JTA  (java Transaction API)

如果事务需要跨多个事务资源,如多个数据库,则需要使用JtaTransactionManager。

<bean  id=”transactionManager”  class=”org.springframework.transaction.jta.JtaTransactionManager”>

<property name=”transactionManagerName” value=”java:/TransactionManager” />

</bean>

JtaTransactionManager将事务委托给一个JTA的实现。JTA规定了应用程序与一个或多个数据源之间协调事务的标准API。transactionManager属性指明了要在JNDI上查找JTA事务管理器。JtaTransactionManager将事务委托给javax.transaction.UserTransaction和javax.transaction.TransactionManager对象。通过UserTransaction.commint()方法来提交事务。反之,调用rollback()。

Spring中的编码事务。

Public void saveSpittle(Spittle spittle){

spitterDao.saveSpittle(spittle);

}

当spittle保存的时候,底层的持久化机制会做很多的事情。

使用spring的TransactionTemplate添加事务性边界(回调机制),

Public void saveSpittle(final Spittle spittle){

txTemplate.execute(new TransactionCallback<void>(){

Public void doInTransaction(TransactionStatus txStatus){

Try{

spitterDao.saveSpittle(spittle);

} catch(RuntimeException e){

txStatus.setRollbackOnly();

throw e;

}

return null;

});

}

使用TransactionTemplate需要实现TransactionCallback接口。因为TransactionCallback接口只有一个要实现的方法,通常很简单的将其实现为匿名内部类,而对于事务性代码,将其放在doInTransaction()方法中。调用TransactionTemplate实例的excute()方法时,将会执行TransactionCallback实例中的代码。若发生异常,则调用TransactionStatus对象的setRollbackOnly回滚事务,反之提交事务。

这里需要在SpitterServiceImpl中注入TransactionTemplate,即

<bean id=”spitterService” class=”#######”

<property  name=”transactionTemplate”

<bean  class=”org.springframework.transaction.support.TransactionTemplate”>

<property name=”transactionManager” ref=”transactionManager” />

</bean>

</property>

</bean>

这里TransactionTemplate注入了一个transactionManager。在背后,TransactionTemplate使用了PlatformTransactionManager实现来处理特定平台的事务管理细节。

Spring声明式事务

Spring对声明式事务的支持是通过spring AOP框架实现的,因为事务实在应用程序主要功能之上的系统级服务,因此可以将spring事务理解成将方法“包装”上事务边界的切面。

Spring提供3种方式来声明事务边界。

即1.使用spring AOP和TtansactionProxyFactoryBean的代理Bean来实现声明式事务;而更好的方式是2.使用Spring的tx命名空间和3.@Transactional注解。

在spring中,声明式事务是通过事务属性来定义的,事务属性描述了事务策略如何应用到方法上,包含5个方面,(1.传播行为,2.隔离级别,3.回滚规则,4.是否只读,5.事务超时)

传播行为回答了这样一个问题,即新的事务应该被启动还是被挂起,或者方法是否要在事务环境中运行

1.PROPAGATION_MANDATORY:

表示该方法必须在事务中运行。若当前事务不存在,则会抛出一个异常。

2.PROPAGATION_NESTED:

表示如果当前已经存在一个事务,那么该方法将会在嵌套事务中运行。嵌套的事务可以独立于当前事务进行单独地提交或回滚。如果当前事务不存在,那么其行为

与PROPAGATION_REQUIRED一样。

3.PROPAGATION_NEVER:

表示当前方法不应该运行在事务上下文中。如果当前正有一个事务在运行,则会抛出异常。

4.PROPAGATION_NOT_SUPPORTED:

表示该方法不应该运行在事务中。如果存在当前事务,在该方法运行期间,当前事务将被挂起。如果使用JTATransactionManager的话,则需要访问TransactionManager。

5.PROPAGATION_REQUIRED:

表示当前方法必须运行在事务中。如果当前事务存在,方法将会在该事务中运行。否则,会启动一个新的事务。

6.PROPAGATION_REQUIRED_NEW:

表示当前方法必须运行在它自己的事务中。一个新的事务将被启动。如果存在当前事务,在该方法运行期间,当前事务被挂起。如果使用JTATransactionManager的话,则需要访问TransactionManager。

7.PROPAGATION_SUPPORTS:

表示当前方法不需要事务上下文,但是如果存在当前事务的话,那么该方法会在这个事务中运行。

隔离级别定义了一个事务可能受其他并发事务影响的程度。

在典型的应用程序中,多个事务并发运行,会操作相同多的数据来完成各自的任务。但是并发可能导致以下问题,即

1.脏读:一个事务读取了另一个事务改写但尚未提交的数据。若改写在稍后被回滚了,那么获取的数据就是无效的。

2.不可重复读:一个事务执行相同的查询两次或以上,但是每次得到的数据不同。通常因为另一个并发事务在两次查询期间更新了数据。

3.。幻读:一个事务读取了几行数据,接着另一个事务插入了一些数据。在随后的查询中,第一个事务发现多了一些原本不存在的记录。

在理想情况下,事务之间是完全隔离的,从而可以防止这些问题发生。但是完全的隔离会导致性能问题,因为它通常会涉及锁定数据库中的记录,甚至整张表。侵占性的锁定会阻碍并发性,要求事务互相等待以完成各自的工作。

1.ISOLATION_DEFAULT:

使用后端数据库默认的隔离级别

2.ISOLATION_READ_UNCOMMITTED:

允许读取尚未提交的数据变更。可能导致脏读、幻读、不可重复读。

3.ISOLATION_READ_COMMITTED:

允许读取并发事务已经提交的数据。仅可以阻止脏读。

4.ISOLATION_REPEATABLE_READ:

对同一字段的多次读取结果是一致的,除非是被本事务自己所修改。仅可以阻止脏读和不可重复读。

5.ISOLATION_SERIALIZABLE:

完全服从ACID隔离级别,确保阻止脏读,不可重复读以及幻读。即最慢的事务隔离级别,其通常是通过锁定事务相关的数据库表来实现的。

只读

如果事务只对后端的数据库进行读操作,数据库可以利用事务的只读特性来进行一些特定的优化。通过将事务设为只读,可以给数据库一个机会,让它应用它认为合适的优化措施。

因为只读优化是在事务启动的时候由数据库实施的,所以只对那些具备启动一个新事务的传播行为(PROPAGATION_REQUIRED、PROPAGATION_REQUIRED_NEW、PROPAGATION_NESTED)的方法来说,才有意义。

事务超时

可以声明一个事务,在特定的秒数后自动回滚,而不是等待其结束。

因为超时时钟会在事务开始时启动,因此同样只对可能启动一个新事务的传播行为有效。

回滚规则

定义哪些异常会导致事务回滚哪些不会。默认情况下,事务只有在遇到运行期异常时才会回滚。

因此事务属性的具体参数定义为:

isolation:指定事务的隔离级别

propagation:定义事务的传播规则

read-only:指定事务是否为只读

rollback-for:指定事务对于哪些检查型异常应当回滚而不提交

no-rollback-for:指定事务对于哪些异常应当继续运行而不回滚

timaout:对于长时间运行的事务定义超长时间。

 

在xml中定义事务

<!-- 使用tx,aop命名空间 -->

<beans xmlns="http://www.springframework.org/schema/beans"

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"

xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context"

xsi:schemaLocation="

http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans-2.0.xsd

http://www.springframework.org/schema/aop

http://www.springframework.org/schema/aop/spring-aop-2.0.xsd

http://www.springframework.org/schema/tx

http://www.springframework.org/schema/tx/spring-tx-2.0.xsd

http://www.springframework.org/schema/context

" >http://www.springframework.org/schema/context/spring-context-2.5.xsd">

<!-- tx命名空间提供一些新的xml配置元素,最重要的为<tx:advice>,其中“transaction-manager”属性指定事务管理器,<tx:method>为指定的方法定义事务参数,可以使用通配符 -->

<tx:advice id="txAdvice" transaction-manager="txManager">

<tx:attributes>

<tx:method name="*" propagation="REQUIRED" isolation="READ_COMMITTED"  read-only="true" />

</tx:attributes>

</tx:advice>

<!-- <tx:advice>只是定义了aop通知,用于把事务边界通知给方法。但这只是事务通知,而不是完整的事务性切面, -->

<!-- 在<tx:advice>中并没有说明哪些bean应该被通知,因此我们需要一个切点来做这件事 -->

<!-- aop命名空间 以下定义一个通知器,它使用txAdvice通知所有实现service接口的bean -->

<!-- 哪些方法应该真正运行在事务中以及方法的事务属性都由这个事务通知即advice-ref来指定,它引用名为txAdvice的通知 -->

<aop:config>

<aop:pointcut id="serviceMethod" expression="execution(* *..service.*())" />

<aop:advisor pointcut-ref="serviceMethod" advice-ref="txAdvice" />

</aop:config>

定义注解驱动事务

除了<tx:advice>元素,tx命名空间还提供了<tx:annotation-driven>元素,这里只需要一行xml:

<tx:annotation-driven transaction-manager=”txManager” />

<tx:annotation-driven>元素告诉spring检查上下文中所有的bean并查找使用@Transactional注解的Bean,而不管这个注解是用在类级别上还是方法级别上已经接口。对于每一个使用@Transactional的Bean,<tx:annotation-driven>会自动为它添加事务通知。通知的事务属性是通过@Transactional注解的参数来定义的。

例:

@Transactional(propagation=Propagation.SUPPORTS,readOnly=true)

Public class SpitterServiceImpl implements SpitterService{

......

@Transactional(propagation=Propagation.SUPPORTS,readOnly=true)

Public void addSpitter(Spitter spitter){

......

}

.......

}

Spring同时支持编码式和声明式的事务管理。不管哪一种,spring都将事务管理平台抽象为通用的API,从而避免直接与特定的事务管理实现打交道。

第6章 事务管理 6.1 spring事务的更多相关文章

  1. Transaction Managament(事务管理二、Spring事务)

    Transaction Managament(事务管理二.Spring事务) Spring事务框架的优势 ​ Spring事务框架将开放过程中事务管理相关的关注点进行了分离,对这些关注点进行了抽象分离 ...

  2. 【Spring】Spring的事务管理 - 1、Spring事务管理概述(数据库事务、Spring事务管理的核心接口)

    Spring事务管理概述 文章目录 Spring事务管理概述 数据库事务 什么是Spring的事务管理? Spring对事务管理的支持 Spring事务管理的核心接口 Platform Transac ...

  3. 事务管理(下) 配置spring事务管理的几种方式(声明式事务)

    配置spring事务管理的几种方式(声明式事务) 概要: Spring对编程式事务的支持与EJB有很大的区别.不像EJB和Java事务API(Java Transaction API, JTA)耦合在 ...

  4. 全面分析 Spring 的编程式事务管理及声明式事务管理

    开始之前 关于本教程 本教程将深入讲解 Spring 简单而强大的事务管理功能,包括编程式事务和声明式事务.通过对本教程的学习,您将能够理解 Spring 事务管理的本质,并灵活运用之. 先决条件 本 ...

  5. Spring学习8-Spring事务管理(编程式事务管理)

    一.Spring事务的相关知识   1.事务是指一系列独立的操作,但在概念上具有原子性. 比如转账:A账号-100, B账号+100,完成.这两个操作独立是没问题的. 但在逻辑上,要么全部完成,要么一 ...

  6. 全面分析 Spring 的编程式事务管理及声明式事务管理--转

    开始之前 关于本教程 本教程将深入讲解 Spring 简单而强大的事务管理功能,包括编程式事务和声明式事务.通过对本教程的学习,您将能够理解 Spring 事务管理的本质,并灵活运用之. 先决条件 本 ...

  7. 框架源码系列十一:事务管理(Spring事务管理的特点、事务概念学习、Spring事务使用学习、Spring事务管理API学习、Spring事务源码学习)

    一.Spring事务管理的特点 Spring框架为事务管理提供一套统一的抽象,带来的好处有:1. 跨不同事务API的统一的编程模型,无论你使用的是jdbc.jta.jpa.hibernate.2. 支 ...

  8. Spring编程式事务管理及声明式事务管理

    本文将深入讲解 Spring 简单而强大的事务管理功能,包括编程式事务和声明式事务.通过对本教程的学习,您将能够理解 Spring 事务管理的本质,并灵活运用之. Spring 事务属性分析 事务管理 ...

  9. 事务之二:spring事务(事务管理方式,事务5隔离级别,7个事务传播行为,spring事务回滚条件)

    事物管理对于企业应用来说是至关重要的,好使出现异常情况,它也可以保证数据的一致性. spring支持编程式事务管理和声明式事务管理两种方式. 编程式事务管理使用TransactionTemplate或 ...

随机推荐

  1. JDBC教程

    JDBC代表Java与数据库的连接,这对Java编程语言和广泛的数据库之间独立于数据库的连接标准的Java API. JDBC库包含的API为每个通常与数据库的使用相关联的任务: 使得连接到数据库 创 ...

  2. 谈谈用Boox Max 2 阅读A4纸文献的体验

    首先说说选择Boox的几个原因: 护眼.这个不用多说,之所以除了电脑,还要电子阅读器,主要是为了护眼. 减少纸质书籍购买.纸质书籍拿在手上是有质感,读起来也更舒服,可一则一些外文书买纸质的是很贵的,相 ...

  3. Asp.Net MVC HttpPost用法

    一个Action只能用一个http 特性,例如:HttpPost 不能与HttpGet 或者多个HttpPost重复使用,否则会出错 也可以用 [AcceptVerbs("put" ...

  4. springboot+kafka+邮件发送(最佳实践)

    导读 集成spring-kafka,生产者生产邮件message,消费者负责发送 引入线程池,多线程发送消息 多邮件服务器配置 定时任务生产消息:计划邮件发送 实现过程 导入依赖 <proper ...

  5. 雪花算法【分布式ID问题】【刘新宇】

    分布式ID 1 方案选择 UUID UUID是通用唯一识别码(Universally Unique Identifier)的缩写,开放软件基金会(OSF)规范定义了包括网卡MAC地址.时间戳.名字空间 ...

  6. 【POJ - 1064】Cable master(二分)

    Cable master Descriptions 输入2个数 N  K n条绳子    要分成大于等于k段 求每段最长多长呢?并且每段不能小于1cm 必须以厘米精度写入数字,小数点后正好是两位数.如 ...

  7. JDK1.8源码分析01之学习建议(可以延伸其他源码学习)

    序言:目前有个计划就是准备看一下源码,来提升自己的技术实力.同时现在好多面试官都喜欢问源码,问你是否读过JDK源码等等? 针对如何阅读源码,也请教了我的老师.下面就先来看看老师的回答,也许会有帮助呢. ...

  8. ajax具体实现学习记录

    记录自己对ajax\的理解, 首先要明白ajax是为了解决什么问题,简单来讲就是为了局部刷新页面,而不刷新整个界面.就比如现在有一个实时热度的显示,它是不断变化的,所以你肯定要不停的从数据库当中获取热 ...

  9. JAVA课堂-动手动脑1

    一.Enum:一般用来表示一组相同类型的常量.对这些属性用常量的好处是显而易见的,不仅可以保证单例,且比较时候可以用”==”来替换equals,枚举对象里面的值都必须是唯一的. 代码: public  ...

  10. 转载:MyBatis mapper.xml中使用静态常量或者静态方法

    转自:https://my.oschina.net/wtslh/blog/682704 今天偶然之间刷到了这样一篇博客,有点意外 mybatis 还可以这样使用ONGL常量的方式,该方式针对 xml的 ...