spring,mybatis事务管理配置与@transactional注解使用 -凯发k8国际

`
zw7534313
  • 浏览: 1232868 次
  • 性别:
  • 来自: 北京
博主相关
  • 博客
  • 微博
  • 相册
  • 收藏
  • 社区版块
    • ( 0)
    • ( 21)
    • ( 1)
    存档分类
    最新评论

    spring,mybatis事务管理配置与@transactional注解使用

     

     

    概述
    事务管理对于企业应用来说是至关重要的,即使出现异常情况,它也可以保证数据的一致性。
    spring framework对事务管理提供了一致的抽象,其特点如下:

    •为不同的事务api提供一致的编程模型,比如jta(java transaction api), jdbc, hibernate, jpa(java persistence api和jdo(java data objects)
    •支持声明式事务管理,特别是基于注解的声明式事务管理,简单易用
    •提供比其他事务api如jta更简单的编程式事务管理api
    •与spring数据访问抽象的完美集成
    事务管理方式

    spring支持编程式事务管理和声明式事务管理两种方式。

    编程式事务管理使用transactiontemplate或者直接使用底层的platformtransactionmanager。对于编程式事务管理,spring推荐使用transactiontemplate。

    声明式事务管理建立在aop之上的。其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。声明式事务最大的优点就是不需要通过编程的方式管理事务,这样就不需要在业务逻辑代码中掺杂事务管理的代码,只需在配置文件中做相关的事务规则声明(或通过基于@transactional注解的方式),便可以将事务规则应用到业务逻辑中。

    显然声明式事务管理要优于编程式事务管理,这正是spring倡导的非侵入式的开发方式。声明式事务管理使业务代码不受污染,一个普通的pojo对象,只要加上注解就可以获得完全的事务支持。和编程式事务相比,声明式事务唯一不足地方是,后者的最细粒度只能作用到方法级别,无法做到像编程式事务那样可以作用到代码块级别。但是即便有这样的需求,也存在很多变通的方法,比如,可以将需要进行事务管理的代码块独立为方法等等。

    声明式事务管理也有两种常用的方式,一种是基于tx和aop名字空间的xml配置文件,另一种就是基于@transactional注解。显然基于注解的方式更简单易用,更清爽。

    自动提交(autocommit)与连接关闭时的是否自动提交

    自动提交

    默认情况下,数据库处于自动提交模式。每一条语句处于一个单独的事务中,在这条语句执行完毕时,如果执行成功则隐式的提交事务,如果
    执行失败则隐式的回滚事务。

    对于正常的事务管理,是一组相关的操作处于一个事务之中,因此必须关闭数据库的自动提交模式。不过,这个我们不用担心,spring会将底层连接的自动提交特性设置为false。
    org/springframework/jdbc/datasource/datasourcetransactionmanager.java


    // switch to manual commit if necessary. this is very expensive in some jdbc drivers,
    // so we don't want to do it unnecessarily (for example if we've explicitly
    // configured the connection pool to set it already).
    if (con.getautocommit()) {
        txobject.setmustrestoreautocommit(true);
        if (logger.isdebugenabled()) {
            logger.debug("switching jdbc connection [" con "] to manual commit");
        }
        con.setautocommit(false);
    }有些数据连接池提供了关闭事务自动提交的设置,最好在设置连接池时就将其关闭。但c3p0没有提供这一特性,只能依靠spring来设置。
    因为jdbc规范规定,当连接对象建立时应该处于自动提交模式,这是跨dbms的缺省值,如果需要,必须显式的关闭自动提交。c3p0遵守这一规范,让客户代码来显式的设置需要的提交模式。

    连接关闭时的是否自动提交

    当一个连接关闭时,如果有未提交的事务应该如何处理?jdbc规范没有提及,c3p0默认的策略是回滚任何未提交的事务。这是一个正确的策略,但jdbc驱动提供商之间对此问题并没有达成一致。
    c3p0的autocommitonclose属性默认是false,没有十分必要不要动它。或者可以显式的设置此属性为false,这样会更明确。

    基于注解的声明式事务管理配置
    spring-servlet.xml





       


    还要在spring-servlet.xml中添加tx名字空间


    ...
        xmlns:tx=""
        xmlns:aop=""
        xsi:schemalocation="
        ...
     

     
     

     
        ...mybatis自动参与到spring事务管理中,无需额外配置,只要org.mybatis.spring.sqlsessionfactorybean引用的数据源与datasourcetransactionmanager引用的数据源一致即可,否则事务管理会不起作用。

    另外需要下载依赖包aopalliance.jar放置到web-inf/lib目录下。否则spring初始化时会报异常
    java.lang.noclassdeffounderror: org/aopalliance/intercept/methodinterceptor

    spring事务特性

    spring所有的事务管理策略类都继承自org.springframework.transaction.platformtransactionmanager接口


    public interface platformtransactionmanager {
     
      transactionstatus gettransaction(transactiondefinition definition)
        throws transactionexception;
     
      void commit(transactionstatus status) throws transactionexception;
     
      void rollback(transactionstatus status) throws transactionexception;
    }其中transactiondefinition接口定义以下特性:

    事务隔离级别

    隔离级别是指若干个并发的事务之间的隔离程度。transactiondefinition 接口中定义了五个表示隔离级别的常量:

    •transactiondefinition.isolation_default:这是默认值,表示使用底层数据库的默认隔离级别。对大部分数据库而言,通常这值就是transactiondefinition.isolation_read_committed。
    •transactiondefinition.isolation_read_uncommitted:该隔离级别表示一个事务可以读取另一个事务修改但还没有提交的数据。该级别不能防止脏读,不可重复读和幻读,因此很少使用该隔离级别。比如postgresql实际上并没有此级别。
    •transactiondefinition.isolation_read_committed:该隔离级别表示一个事务只能读取另一个事务已经提交的数据。该级别可以防止脏读,这也是大多数情况下的推荐值。
    •transactiondefinition.isolation_repeatable_read:该隔离级别表示一个事务在整个过程中可以多次重复执行某个查询,并且每次返回的记录都相同。该级别可以防止脏读和不可重复读。
    •transactiondefinition.isolation_serializable:所有的事务依次逐个执行,这样事务之间就完全不可能产生干扰,也就是说,该级别可以防止脏读、不可重复读以及幻读。但是这将严重影响程序的性能。通常情况下也不会用到该级别。
    事务传播行为

    所谓事务的传播行为是指,如果在开始当前事务之前,一个事务上下文已经存在,此时有若干选项可以指定一个事务性方法的执行行为。在transactiondefinition定义中包括了如下几个表示传播行为的常量:

    •transactiondefinition.propagation_required:如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。这是默认值。
    •transactiondefinition.propagation_requires_new:创建一个新的事务,如果当前存在事务,则把当前事务挂起。
    •transactiondefinition.propagation_supports:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
    •transactiondefinition.propagation_not_supported:以非事务方式运行,如果当前存在事务,则把当前事务挂起。
    •transactiondefinition.propagation_never:以非事务方式运行,如果当前存在事务,则抛出异常。
    •transactiondefinition.propagation_mandatory:如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。
    •transactiondefinition.propagation_nested:如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行;如果当前没有事务,则该取值等价于transactiondefinition.propagation_required。
    事务超时

    所谓事务超时,就是指一个事务所允许执行的最长时间,如果超过该时间限制但事务还没有完成,则自动回滚事务。在 transactiondefinition 中以 int 的值来表示超时时间,其单位是秒。

    默认设置为底层事务系统的超时值,如果底层数据库事务系统没有设置超时值,那么就是none,没有超时限制。

    事务只读属性

    只读事务用于客户代码只读但不修改数据的情形,只读事务用于特定情景下的优化,比如使用hibernate的时候。
    默认为读写事务。

    spring事务回滚规则

    指示spring事务管理器回滚一个事务的推荐方法是在当前事务的上下文内抛出异常。spring事务管理器会捕捉任何未处理的异常,然后依据规则决定是否回滚抛出异常的事务。

    默认配置下,spring只有在抛出的异常为运行时unchecked异常时才回滚该事务,也就是抛出的异常为runtimeexception的子类(errors也会导致事务回滚),而抛出checked异常则不会导致事务回滚。
    可以明确的配置在抛出那些异常时回滚事务,包括checked异常。也可以明确定义那些异常抛出时不回滚事务。

    还可以编程性的通过setrollbackonly()方法来指示一个事务必须回滚,在调用完setrollbackonly()后你所能执行的唯一操作就是回滚。

    @transactional注解

    @transactional属性

     

      属性 类型 描述
    value string 可选的限定描述符,指定使用的事务管理器
    propagation enum: propagation 可选的事务传播行为设置
    isolation enum: isolation 可选的事务隔离级别设置
    readonly boolean 读写或只读事务,默认读写
    timeout int (in seconds granularity) 事务超时时间设置
    rollbackfor class对象数组,必须继承自throwable 导致事务回滚的异常类数组
    rollbackforclassname 类名数组,必须继承自throwable 导致事务回滚的异常类名字数组
    norollbackfor class对象数组,必须继承自throwable 不会导致事务回滚的异常类数组
    norollbackforclassname 类名数组,必须继承自throwable 不会导致事务回滚的异常类名字数组

    用法

    @transactional 可以作用于接口、接口方法、类以及类方法上。当作用于类上时,该类的所有 public 方法将都具有该类型的事务属性,同时,我们也可以在方法级别使用该标注来覆盖类级别的定义。

    虽然 @transactional 注解可以作用于接口、接口方法、类以及类方法上,但是 spring 建议不要在接口或者接口方法上使用该注解,因为这只有在使用基于接口的代理时它才会生效。另外, @transactional 注解应该只被应用到 public 方法上,这是由 spring aop 的本质决定的。如果你在 protected、private 或者默认可见性的方法上使用 @transactional 注解,这将被忽略,也不会抛出任何异常。

    默认情况下,只有来自外部的方法调用才会被aop代理捕获,也就是,类内部方法调用本类内部的其他方法并不会引起事务行为,即使被调用方法使用@transactional注解进行修饰。


    @transactional(readonly = true)
    public class defaultfooservice implements fooservice {
     
      public foo getfoo(string fooname) {
        // do something
      }
     
      // these settings have precedence for this method
      //方法上注解属性会覆盖类注解上的相同属性
      @transactional(readonly = false, propagation = propagation.requires_new)
      public void updatefoo(foo foo) {
        // do something
      }
    }

     

    事物传播行为介绍: 
    @transactional(propagation=propagation.required) 
    如果有事务, 那么加入事务, 没有的话新建一个(默认情况下)
    @transactional(propagation=propagation.not_supported) 
    容器不为这个方法开启事务
    @transactional(propagation=propagation.requires_new) 
    不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务
    @transactional(propagation=propagation.mandatory) 
    必须在一个已有的事务中执行,否则抛出异常
    @transactional(propagation=propagation.never) 
    必须在一个没有的事务中执行,否则抛出异常(与propagation.mandatory相反)
    @transactional(propagation=propagation.supports) 
    如果其他bean调用这个方法,在其他bean中声明事务,那就用事务.如果其他bean没有声明事务,那就不用事务.

    事物超时设置:
    @transactional(timeout=30) //默认是30秒

    事务隔离级别:
    @transactional(isolation = isolation.read_uncommitted)
    读取未提交数据(会出现脏读, 不可重复读) 基本不使用
    @transactional(isolation = isolation.read_committed)
    读取已提交数据(会出现不可重复读和幻读)
    @transactional(isolation = isolation.repeatable_read)
    可重复读(会出现幻读)
    @transactional(isolation = isolation.serializable)
    串行化

    mysql: 默认为repeatable_read级别
    sqlserver: 默认为read_committed

    脏读 : 一个事务读取到另一事务未提交的更新数据
    不可重复读 : 在同一事务中, 多次读取同一数据返回的结果有所不同, 换句话说, 
    后续读取可以读到另一事务已提交的更新数据. 相反, "可重复读"在同一事务中多次
    读取数据时, 能够保证所读数据一样, 也就是后续读取不能读到另一事务已提交的更新数据
    幻读 : 一个事务读到另一个事务已提交的insert数据

     


     
      
     

     
     
     

    分享到:
    |
    评论

    相关推荐

      struts2、spring与mybatis整合 包含简单登录功能 事务配置,运行junittestvillagearticle的modify测试。配置说明参考注释

      很好的实例,简单易懂,适合新手

      本文处理spring mybatis,或springjdbc,对oracle事务不能提交的问题分析与解决

      spring 与 各框架的组各下来,版本就特别的多,针对简单的 使用注解来管理事务的,研究了两天,要不网上写的不清楚,要不版本都很旧了,所以就是不回滚,spring 的配置太灵活了,在加上新手,根本就不可能明白spring...

      springmvc spring mybatis maven 注解方式

      mybatis和spring整合方式-带注解,个人经验总结,适合新手学习,欢迎下载

      mybatis的注解方式工程,导入resource下的mysql创建数据库即可运行,含有一对一模式 pringcloud mybatis的注解方式的代码,只有数据库配置文件,mybatis无需mapper.xml,全注解方式

      此工具自动生成非注解时spring整合mybatis时需要的一些配置文件,对于使用较新版本的spring,可以忽视生成的dao的配置文件,建议采用spring的扫描方式。生成的generatorconfigxml修改jdbc驱动位置后可直接用使用...

      mybatis-plus中乐观锁@version注解的问题与凯发k8国际娱乐官网入口的解决方案.docx

      spring mvc 和 mybatis 整合、 mybatis使用注解模式

      spring mybatis springmvc maven框架,单模块,导入就可以使用,里面有一个user的增删查改,大家拿来修改就可以使用

      使用环境:myeclipse/eclipse tomcat mysql。 使用技术:spring mvc spring mybatis / jsp servlet javabean jdbc。

      spring3.0mvc自学教程ppt 注解教程 注解实例 springmybatis教程 项目实例 中文api 初学(自学)着的不二法定,从入门到项目实例深入学习(浅-深)

      maven spring mybatis mysql 事务管理 企业级开发

      struts spring mybatis easyui(jquery)注解案例。采用注解的方式跟踪代码依赖性,实现替代配置文件功能。里面包含了平时用的增删改查及分页,分页查询时用的是存储过程,提高了数据库的性能。代码简单、易懂。

      mybatis3.2.3整合spring3.2.4使用存注解实现

      springmvc spring mybatis 框架整合需要的jar包 配置文件 工具类

      struts2springmybatis 增删改查,里面的struts2,spring,mybatis配置文件都是模块化的管理,实现了社会版本的配置,不再是个人玩弄的小项目。具体的 sql语句也已经加到了webrrot目录下面了,自己下载下来可以详细看...

      花了一晚上时间终于搞出来了spring整合mybatis事务回滚(mysql数据库),控制service层中的多次插入操作,多次操作整体是一个事务。 里面有缘嘛和jar包,资源为war包,导入即可。运行test类中的测试代码即可。 建表...

    global site tag (gtag.js) - google analytics
    网站地图