发现问题

最近,碰到一个问题,再用spring实现事务管理的时候,发现不起作用,在出异常时,并不会回滚数据库操作。

我想实现的功能如下:

  1. @Transactional(isolation=Isolation.DEFAULT,readOnly=false,propagation=Propagation.REQUIRED,rollbackFor=Exception.class)
  2. public boolean someService() {
  3. mapper.insert();
  4. mapper.insert();
  5. }

TransactionManager的配置如下:

  1. <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  2. <property name="dataSource" ref="dataSource" />
  3. </bean>

但是在测试中,在service中手动抛出异常后,第一条insert语句并不会回滚。

之后研究了很久,在控制台打印的消息显示,并没有开启事务。那么我分析应该有下面几种可能:

  1. 配置错误
  2. JDBC的AutoCommit没有设置为false
  3. 控制台信息显示,在Service中,insertA在jdbt连接池中取出了一个连接,插入完后,便将连接返回给了连接池,所以在service中,有多个数据库连接分别处理不同的insert语句。这也可能导致事务失败。

下面来逐步排除:

AutoCommit=true,这个是有可能的,但是通过查看Spring的源码,发现Spring在事务中,是会检测当前连接的AutoCommit选项并把它设置为False,所以排除

最后,通过查看Spring关于Transaction的文档,发现上面说了,如果使用的是DataSourceTransactionManager,那么只能在一个数据库连接上开启事务。那么这么说,应该是数据库连接池的问题了?或者是Spring自己并没有开启事务管理,而是将事务管理委托给了Mysql去实现?这个是有可能的,所以后面再看。

解决问题

然后,我选择使用了切面Aop来实现Spring的事务管理。步骤如下:

  1. //注入transactionManager
  2. @Autowired() @Qualifier("transactionManager1")
  3. private PlatformTransactionManager transactionManager;

这一步是获得自动注入的 transactionManager实例,之后使用TransactionTemplate

  1. //编程式开启事务
  2. TransactionTemplate template = new TransactionTemplate(transactionManager);

是不是很像JDBCTemplate?那么一样的使用内部类实现回调:

  1. //在这边也可以取消返回值,设为void
  2. boolean result = (Boolean) template.execute(new TransactionCallback<Object>() {
  3. public Object doInTransaction(TransactionStatus transactionStatus) {
  4. try {
  5. mapper.insertA();
  6. throw new SQLException();
  7. mapper.insertB();
  8. } catch (Exception e) {mapper.insertB();
  9. //出异常时,手动设置回滚
  10. transactionStatus.setRollbackOnly();
  11. e.printStackTrace();
  12. return false;
  13. }
  14. return true;
  15. }
  16. });

这样,就可以使用Aop手动开启事务了。

经过测试,在两个插入之间手动抛出了异常,insertA的插入被回滚了。从控制台输出信息来看,insertA插入成功后,即把jdbc返回到连接池了,并且事务管理配置的依然是DataSourceTransactionManager,说明前面的数据库连接池的问题被排除了。

分析原因

那么就剩下配置错误了。

实验多次以后,我发现如下配置是可以让声明式事务管理起作用的:

Spring的配置中,取消注入Controller

  1. <context:component-scan base-package="com.test">
  2. <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller" />
  3. </context:component-scan>

SpringMVC的配置,取消注入Service

  1. <context:component-scan base-package="com.test">
  2. <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller" />
  3. <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Service" />
  4. </context:component-scan>

配置事务的Service实现某个接口

  1. @Service
  2. public class Test() {
  3. @Transactional(isolation=Isolation.DEFAULT,readOnly=false,propagation=Propagation.REQUIRED,rollbackFor=Exception.class)
  4. public boolean someService() {
  5. mapper.insert();
  6. mapper.insert();
  7. }
  8. }

这样便可以开启事务,并在遇到异常时回滚了。那为什么原来的配置就不行呢?

根本原因是因为spring的context是父子容器,由ServletContextListener 加载spring配置文件产生的是父容器,springMVC加载配置文件产生的是子容器,子容器对Controller进行扫描装配时装配了@Service注解的实例 (@Controller 实例依赖@Service实例),而该实例理应由父容器进行初始化以保证事务的增强处理,所以此时得到的将是原样的Service,没有经过事务加强处理,故而没有事务处理能力

所以在spring的配置中我们要手动exclude controller的注入,不扫描带有@Controller注解的类。因为这些类已经随容器启动时,在servlet-context中扫描过一遍了

在springMVC的配置中扫描业务组件,让springMVC不扫描带有@Service注解的类(留在spring中扫描@Service注解的类),防止事务失效。

在需要配置事务管理的Service类中,需要让类继承一个或几个接口,是因为如果不继承接口,那么会使用cglib动态代理去实现,那么事务管理也会失效。如果实现了接口,那么便会使用jdk自带的动态代理去实现,这样便可以成功的开启事务了。

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. cas配置全攻略(转)

    转:http://www.blogjava.net/tufanshu/archive/2011/01/21/343290.html 经过将近两天的测试,参考众多网友的贡献,终于完成了对cas的主要配置 ...

  2. 卸载mysql时,如何卸载干净!

    相信很多朋友在使用mysql的过程中都会遇到这样的问题,安装过程出错,或者想要换个版本,或者不想使用了,这个时候我们都需要完全卸载mysql呢?下面,就来谈一谈我的经验. 1.控制面板——>所有 ...

  3. hdu1102 Constructing Roads (简单最小生成树Prim算法)

    Problem Description There are N villages, which are numbered from 1 to N, and you should build some ...

  4. extjs中datefield组件的使用

    xtype: 'datefield', id: 'dateShangmfa', name: 'dateShangmfa', fieldLabel: '日期',//设置标签文本 editable: fa ...

  5. ECharts 是一款开源

    ECharts

  6. sublime text3 下搭建python IDE环境 --Anaconda插件篇

    近来在园区发现大家使用ST3(Sublime text 3)工具进行python学习.然后自己也跟风了一把. 1.ST3下载地址: http://www.sublimetext.com/3 2.安装S ...

  7. 一起学习iOS开发专用词汇,每天记3个,助你变大牛-b

    大家做开发最大的问题是什么?英语的问题应该困扰很多的同学的地方,我们提倡科学学习开发中的常用词汇.我们不要求大家有特别好的听.说.写,只要能够记住,能够认识这些常用词汇你以后的开发也将游刃有余.我们的 ...

  8. matplotlib绘制精美的图表(这是教程哦)

    http://sebug.net/paper/books/scipydoc/matplotlib_intro.html

  9. javascript language

    function class(function, this, prototype) closure与function expression没有任何关系. closure必然与function联系在一起 ...

  10. vijosP1499炸毁燃料库

    背景 某天,外星人展开了对地球的侵略,OIer们开始与之周旋... 描述 外星人派出了172849个外星人乘着UFO来到地球,curimit神new带领着OIer们奋力抵抗.curimit神new觉得 ...