`

mybatis 、jdbc 、 spring事务模板

    博客分类:
  • SSIH
 
阅读更多
  
mybatis深入:
http://blog.csdn.net/column/details/mybatis-principle.html




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


<!-- transaction support-->
<!-- PlatformTransactionMnager -->
<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource" />
</bean>
<!-- enable transaction annotation support -->
<tx:annotation-driven transaction-manager="txManager" />还要在spring-servlet.xml中添加tx名字空间


...
    xmlns:tx="http://www.springframework.org/schema/tx"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="
    ...

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


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

    ...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
  }
}
分享到:
评论

相关推荐

    MyBatis与Spring整合jar包(1).rar

    开发者可以将这些库导入到自己的项目中,根据提供的配置模板快速设置好MyBatis与Spring的整合。 总的来说,MyBatis与Spring的整合简化了Java开发中的数据访问层,提供了更好的事务管理和依赖注入,使得项目结构更加...

    MyBatis和Spring整合jar:mybatis-3.2.7.jar等

    这个jar包提供了一些关键组件,如SqlSessionTemplate和SqlSessionDaoSupport,它们是Spring的模板模式实现,可以帮助开发者在Spring环境下方便地使用MyBatis。此外,它还提供了MapperScannerConfigurer,可以自动...

    mybatis-spring整合项目

    2. **事务管理**: Spring可以控制MyBatis的事务边界,通过PlatformTransactionManager实现事务的提交和回滚。 3. **MapperScannerConfigurer**: 这是一个Spring配置类,用于扫描指定包下的Mapper接口,自动注册为...

    spring_mybatis_spring-mybatis_

    Spring MyBatis 模板是为初学者设计的一个基础项目框架,它整合了Spring和MyBatis两大流行框架,帮助开发者快速搭建一个可运行的Web应用。本文将深入讲解Spring MyBatis集成的关键知识点,包括Spring框架的核心概念...

    一个基础mybatis-spring程序所需的jar包

    7. **org.springframework.jdbc-3.1.3.RELEASE.jar**:Spring的JDBC抽象层,提供了模板方法和数据源管理,简化了数据库操作,同时也支持事务管理。 8. **org.springframework.aop-3.1.3.RELEASE.jar**:Spring的AOP...

    mybatis mybatis-spring source jar

    MyBatis-Spring 可以帮助开发者无缝地将 MyBatis 集成到 Spring 应用中,使得事务管理、依赖注入等 Spring 的特性得以充分利用。 在源码分析中,`mybatis-3.1.1-sources.jar` 包含了 MyBatis 3.1.1 版本的全部源...

    mybatis3与Spring整合官方中文版.pdf

    `SqlSessionTemplate`是MyBatis-Spring提供的一个模板类,用于执行SQL操作。它实现了`SqlSessionOperations`接口,提供了一系列方便的方法来操作数据库。 **5.2 SqlSessionSupport** `SqlSessionSupport`是另一个...

    mybatis+mybatis-spring+mybatis-generator

    MyBatis是一个优秀的Java持久层框架,它支持...例如,MyBatis Generator可以快速生成基础代码,MyBatis提供灵活的SQL映射,而MyBatis-Spring则帮助整合Spring的依赖注入和事务管理,使得整个系统更加健壮且易于测试。

    mybatis-spring-1.2.xsd

    在实际项目中,结合Spring的其他功能,如AOP、JDBC模板、事务管理等,可以构建出健壮、灵活的后端服务。 总结来说,`mybatis-spring-1.2.xsd`是MyBatis与Spring集成的关键,它定义了在Spring XML配置文件中声明和...

    Mybatis spring spring mvc

    在整合Mybatis、Spring和Spring MVC...总之,Mybatis、Spring和Spring MVC的集成涉及多个配置文件的整合,事务管理的设置,以及各层组件之间的协调。正确配置和理解这些知识点对于构建高效、可维护的Web应用至关重要。

    spring-mybatis-spring-3.0.1.zip

    而MyBatis则专注于SQL的编写和执行,避免了传统的JDBC中的大量模板代码,提高了开发效率。两者的整合,可以让Spring管理MyBatis的SqlSession和Mapper,实现更优雅的数据访问控制。 二、整合步骤 1. 引入依赖:在...

    MyBatis+Spring3整合

    MyBatis 和 Spring3 的整合是一项常见的任务,特别是在构建企业级应用时,为了利用 MyBatis 的灵活性和 Spring 的依赖注入以及事务管理能力。在早期版本的 Spring 中,可能存在对最新 MyBatis 版本的支持不足的问题...

    mybatis-spring-1.0.0-RC1-bundle.zip

    通过 MyBatis-Spring,你可以充分利用 Spring 的优点,如事务管理、依赖注入,同时享受 MyBatis 的灵活性和直观的 SQL 编写方式。这个集成库简化了在 Spring 应用中使用 MyBatis 的过程,使得开发者可以更专注于业务...

    mybatis与spring整合

    Spring还提供了多种数据访问集成,包括JDBC模板、ORM框架集成(如Hibernate和MyBatis)等。 MyBatis与Spring整合的主要目的是实现更高效的事务管理。Spring通过PlatformTransactionManager接口来管理事务,它可以...

    Mybatis+spring整合小例子--亲自做的可以运行

    总结来说,Mybatis+Spring的整合涉及到的主要知识点包括:Mybatis的配置与SqlSession管理、Spring的bean管理和依赖注入、Mapper接口与XML的映射、事务管理。这个小例子提供了一个基础的模板,让初学者能够快速上手...

    springboot+redis+mybatis事务和引擎模板

    - 在MyBatis中,事务与Spring事务管理结合,可以确保数据库操作在同一个事务中完成。 6. **模板引擎**: - 尽管标题中提到“引擎模板”,但在Spring Boot中常见的模板引擎有Thymeleaf、Freemarker等,它们用于...

    MyBatis+Spring的增、删、查、改以及分页和事务管理

    通过XML或注解配置,MyBatis允许开发者编写自定义的SQL,存储过程,甚至完整的SQL逻辑,从而避免了传统的JDBC中的大量模板代码。 Spring框架则是一个全面的企业级应用开发框架,提供依赖注入(DI)、面向切面编程...

    SSM中spring与mybatis整合

    总结起来,Spring与MyBatis的整合主要涉及到Spring对MyBatis的SqlSessionFactory和Mapper接口的管理,以及事务控制。通过这种方式,开发者可以充分利用Spring的IoC和AOP特性,同时享受MyBatis带来的简单灵活的SQL...

    Mybatis+Spring整合简单实例

    总的来说,这个实例提供了一个基础的Mybatis和Spring整合的模板,对于初学者来说,这是一个很好的起点,可以帮助他们理解如何在实际项目中将这两个强大的框架结合起来。通过这个实例,你可以学习到Spring如何管理...

    spring-mybatis-spring-1.0.0-RC2.zip

    MyBatis则是一个基于Java的持久层框架,它封装了JDBC,让开发者可以更专注于SQL语句的编写,同时通过XML或注解方式将SQL语句与Java代码解耦。MyBatis的主要特点是它的动态SQL支持,使得SQL的编写更加灵活。 整合...

Global site tag (gtag.js) - Google Analytics