spring是SSH中的管理员,负责管理其它框架,协调各个部分的工作。今天一起学习一下Spring的事务管理。Spring配置文件中关于事务配置总是由三个组成部分,分别是DataSource、TransactionManager和代理机制这三部分,无论哪种配置方式,一般变化的只是代理机制这部分。 DataSource、TransactionManager这两部分只是会根据数据访问方式有所变化,比如使用hibernate进行数据访问时,DataSource实际为SessionFactory,TransactionManager的实现为HibernateTransactionManager。 下面一起看看三种声明式事务的具体配置:

 

公共配置

  1. <!-- 配置sessionFactory -->
  2. <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
  3. <property name="configLocation">
  4. <value>classpath:config/hibernate.cfg.xml</value>
  5. </property>
  6. <property name="packagesToScan">
  7. <list>
  8. <value>com.entity</value>
  9. </list>
  10. </property>
  11. </bean>
  12. <!-- 配置事务管理器(声明式的事务) -->
  13. <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
  14. <property name="sessionFactory" ref="sessionFactory"></property>
  15. </bean>
  16. <!-- 配置DAO -->
  17. <bean id="userDao" class="com.dao.UserDaoImpl">
  18. <property name="sessionFactory" ref="sessionFactory"></property>
  19. </bean>

第一种,使用tx标签方式

  1. <!-- 第一种配置事务的方式 ,tx-->
  2. <tx:advice id="txadvice" transaction-manager="transactionManager">
  3. <tx:attributes>
  4. <tx:method name="add*" propagation="REQUIRED" rollback-for="Exception" />
  5. <tx:method name="modify*" propagation="REQUIRED" rollback-for="Exception" />
  6. <tx:method name="del*" propagation="REQUIRED" rollback-for="Exception"/>
  7. <tx:method name="*" propagation="REQUIRED" read-only="true"/>
  8. </tx:attributes>
  9. </tx:advice>
  10. <aop:config>
  11. <aop:pointcut id="daoMethod" expression="execution(* com.dao.*.*(..))"/>
  12. <aop:advisor pointcut-ref="daoMethod" advice-ref="txadvice"/>
  13. </aop:config>

expression="execution(* com.dao.*.*(..))"
其中第一个*代表返回值,第二*代表dao下子包,第三个*代表方法名,“(..)”代表方法参数。

第二种,使用代理方式

  1. <!-- 第二种配置事务的方式 ,代理-->
  2. <bean id="transactionProxy"
  3. class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean" abstract="true">
  4. <property name="transactionManager" ref="transactionManager"></property>
  5. <property name="transactionAttributes">
  6. <props>
  7. <prop key="add*">PROPAGATION_REQUIRED, -Exception</prop>
  8. <prop key="modify*">PROPAGATION_REQUIRED, -Exception</prop>
  9. <prop key="del*">PROPAGATION_REQUIRED, -Exception</prop>
  10. <prop key="*">PROPAGATION_REQUIRED, readOnly</prop>
  11. </props>
  12. </property>
  13. </bean>
  14. <bean id="userDao" parent="transactionProxy">
  15. <property name="target">
  16. <!-- 用bean代替ref的方式-->
  17. <bean class="com.dao.UserDaoImpl">
  18. <property name="sessionFactory" ref="sessionFactory"></property>
  19. </bean>
  20. </property>
  21. </bean>

将transactionProxy的abstract属性设置为"true",然后将具体的Dao的parent属性设置为"transactionProxy",可以精简代码。

第三种,使用拦截器

  1. <!-- 第三种配置事务的方式,拦截器 (不常用)-->
  2. <bean id="transactionInterceptor" class="org.springframework.transaction.interceptor.TransactionInterceptor">
  3. <property name="transactionManager" ref="transactionManager"></property>
  4. <property name="transactionAttributes">
  5. <props>
  6. <prop key="add*">PROPAGATION_REQUIRED, -Exception</prop>
  7. <prop key="modify*">PROPAGATION_REQUIRED, -Exception</prop>
  8. <prop key="del*">PROPAGATION_REQUIRED, -Exception</prop>
  9. <prop key="*">PROPAGATION_REQUIRED, readOnly</prop>
  10. </props>
  11. </property>
  12. </bean>
  13. <bean id="proxyFactory" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
  14. <property name="interceptorNames">
  15. <list>
  16. <value>transactionInterceptor</value>
  17. </list>
  18. </property>
  19. <property name="beanNames">
  20. <list>
  21. <value>*Dao</value>
  22. </list>
  23. </property>
  24. </bean>

Spring事务类型详解:

PROPAGATION_REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。

PROPAGATION_SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。

PROPAGATION_MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。

PROPAGATION_REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。

PROPAGATION_NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。

PROPAGATION_NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。

PROPAGATION_NESTED--如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作。

<tx:method/> 有关的设置

属性 是否需要? 默认值 描述
name  

与事务属性关联的方法名。通配符(*)可以用来指定一批关联到相同的事务属性的方法。如:'get*''handle*''on*Event'等等。

propagation REQUIRED 事务传播行为
isolation DEFAULT 事务隔离级别
timeout -1 事务超时的时间(以秒为单位)
read-only false 事务是否只读?
rollback-for  

将被触发进行回滚的 Exception(s);以逗号分开。 如:'com.foo.MyBusinessException,ServletException'

no-rollback-for  

 被触发进行回滚的 Exception(s);以逗号分开。 如:'com.foo.MyBusinessException,ServletException'

采用注解的方式,需要注意的是,使用注解的方式需要在Spring的配置文件中加入一句话:<context:annotation-config />,其作用是开启注解的方式。具体配置如下:

  1. <!--开启注解方式-->
  2. <context:annotation-config />
  3. <!-- 配置sessionFactory -->
  4. <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
  5. <property name="configLocation">
  6. <value>classpath:config/hibernate.cfg.xml</value>
  7. </property>
  8. <property name="packagesToScan">
  9. <list>
  10. <value>com.entity</value>
  11. </list>
  12. </property>
  13. </bean>
  14. <!-- 配置事务管理器 -->
  15. <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
  16. <property name="sessionFactory" ref="sessionFactory"></property>
  17. </bean>
  18. <!-- 第四种配置事务的方式,注解 -->
  19. <tx:annotation-driven transaction-manager="transactionManager"/>

注解文件:

  1. package com.dao;
  2. import org.springframework.orm.hibernate3.HibernateTemplate;
  3. import org.springframework.transaction.annotation.Propagation;
  4. import org.springframework.transaction.annotation.Transactional;
  5. import com.entity.User;
  6. @Transactional
  7. public class UserDaoImpl_BAK extends HibernateTemplate {
  8. @Transactional(propagation=Propagation.REQUIRED,rollbackForClassName="Exception")
  9. public void addUser(User user) throws Exception {
  10. this.save(user);
  11. }
  12. @Transactional(propagation=Propagation.REQUIRED,rollbackForClassName="Exception")
  13. public void modifyUser(User user) {
  14. this.update(user);
  15. }
  16. @Transactional(propagation=Propagation.REQUIRED,rollbackForClassName="Exception")
  17. public void delUser(String username) {
  18. this.delete(this.load(User.class, username));
  19. }
  20. @Transactional(readOnly=true)
  21. public void selectUser() {
  22. }
  23. }

类头的@Transactional为默认事务配置,如方法没有自己的事务类型,则按默认事务,如有自己的配置,则按自己的配置。

以上四种配置方式最常用的还是第一、二种,第三种是比较老旧的方式,而注解的方式不太适合比较大的项目,用于简单的小项目还是很好的,其特点就是简单明了。每种方法都有每种方法的特点跟适用的环境,没有绝对的好与坏,只不过前两种在实际的工作当中用的更多一些。

转:Spring事务管理的更多相关文章

  1. 【Java EE 学习 52】【Spring学习第四天】【Spring与JDBC】【JdbcTemplate创建的三种方式】【Spring事务管理】【事务中使用dbutils则回滚失败!!!??】

    一.JDBC编程特点 静态代码+动态变量=JDBC编程. 静态代码:比如所有的数据库连接池 都实现了DataSource接口,都实现了Connection接口. 动态变量:用户名.密码.连接的数据库. ...

  2. spring事务管理器设计思想(二)

    上文见<spring事务管理器设计思想(一)> 对于第二个问题,涉及到事务的传播级别,定义如下: PROPAGATION_REQUIRED-- 如果当前没有事务,就新建一个事务.这是最常见 ...

  3. spring事务管理器设计思想(一)

    在最近做的一个项目里面,涉及到多数据源的操作,比较特殊的是,这多个数据库的表结构完全相同,由于我们使用的ibatis框架作为持久化层,为了防止每一个数据源都配置一套规则,所以重新实现了数据源,根据线程 ...

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

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

  5. Spring事务管理器的应对

    Spring抽象的DAO体系兼容多种数据访问技术,它们各有特色,各有千秋.像Hibernate是非常优秀的ORM实现方案,但对底层SQL的控制不太方便:而iBatis则通过模板化技术让你方便地控制SQ ...

  6. Spring事务管理(转)

    1 初步理解 理解事务之前,先讲一个你日常生活中最常干的事:取钱. 比如你去ATM机取1000块钱,大体有两个步骤:首先输入密码金额,银行卡扣掉1000元钱:然后ATM出1000元钱.这两个步骤必须是 ...

  7. [Spring框架]Spring 事务管理基础入门总结.

    前言:在之前的博客中已经说过了数据库的事务, 不过那里面更多的是说明事务的一些锁机制, 今天来说一下Spring管理事务的一些基础知识. 之前的文章: [数据库事务与锁]详解一: 彻底理解数据库事务一 ...

  8. Spring 事务管理 01 ——

    目录: 参考: 1.Spring 事务管理高级应用难点剖析: 第 1 部分

  9. Spring 事务管理原理探究

    此处先粘贴出Spring事务需要的配置内容: 1.Spring事务管理器的配置文件: 2.一个普通的JPA框架(此处是mybatis)的配置文件: <bean id="sqlSessi ...

  10. Spring 事务管理高级应用难点剖析--转

    第 1 部分 http://www.ibm.com/search/csass/search/?q=%E4%BA%8B%E5%8A%A1&sn=dw&lang=zh&cc=CN& ...

随机推荐

  1. 使用Typescript重构axios(六)——实现基础功能:获取响应数据

    0. 系列文章 1.使用Typescript重构axios(一)--写在最前面 2.使用Typescript重构axios(二)--项目起手,跑通流程 3.使用Typescript重构axios(三) ...

  2. vue-cli3.X快速创建项目

    1.安装 Vue CLI 的包名称由 vue-cli 改成了 @vue/cli. 如果你已经全局安装了旧版本的 vue-cli (1.x 或 2.x),你需要先通过以下方式先卸载它: npm unin ...

  3. JavaScript-原型对象与原型链

    原型对象 1.每个对象一定会有一个原型对象 2.原型对象实际是构造实例对象的构造器中的一个属性,只不过这个属性是个对象 3.这个原型对象中的属性与方法,都会被对象实例所共享(类似python中的类方法 ...

  4. canvas线条实践之运动的正方形

    原理说明: 1.通过rect实现正方形的绘制: 2.save保存canvas面板的保存,restore回复保存的canvas面板到初始状态: 3.translate用于改变canvas坐标的起始位置: ...

  5. 为什么我加了索引,SQL执行还是这么慢(二)?

    接上文 在MySQL中,有一些语句即使逻辑相同,执行起来的性能差异确实极大的. 还记得我们上文中的结论吗:如果想使用索引树搜索功能,就不能使用数据库函数来处理索引字段值,而是在不改变索引字段值的同时, ...

  6. Ansibile之playbook初识

    一.playbook简介 ansiblie的任务配置文件被称为playbook,俗称“剧本”,每一个剧本(playbook)中都包含了一系列的任务,这每个任务在ansible中又被称为“戏剧”(pla ...

  7. 那些年用过的UI开发平台

    屈指算来,在我不长也不能算短的职业生涯中,接触了数代 的UI技术: MFC (Microsoft Foundation Class)- Win32上最强大的Class Library,没有之一.VS唯 ...

  8. netty源码解析(4.0)-29 Future模式的实现

    Future模式是一个重要的异步并发模式,在JDK有实现.但JDK实现的Future模式功能比较简单,使用起来比较复杂.Netty在JDK Future基础上,加强了Future的能力,具体体现在: ...

  9. gdb(ddd,kdevelop等)调试ZeroIce开发的应用程序,中断信号引起的问题

    不作文,只记要点. 1.Ice::Application的程序框架默认对SIGHUP, SIGINT, SIGTERM进行处理.目的就是捕捉Ctrl+C发出信号有序地结束程序.这个功能扰乱了我们使用g ...

  10. nginx-(三)基本模块1

    nginx常用模块介绍 ngx_http_access_module模块的配置(基于IP的访问控制) allow address | CIDR | unix: | all; deny address ...