资讯专栏INFORMATION COLUMN

MySQL 原生SQL、JDBC、Spring事务使用

Euphoria / 2662人阅读

摘要:也不需要增加对事务设计学习的成本。原生事务开启方式注意本身是不支持嵌套事务的。在中开启一个事务后,如果在为提交前又开启了一个事务,则会自动提交当前事务。所以这种事务的有效范围也只限于当前请求中的一个。

MySQL事务通过简单的Sql语句就可以开启提交或回滚事务,实际使用中特别是Spring对事务的封装,可以让我们通过声明或注解的方式就可以控制事务,还可以支持事务的嵌套、传播。为了理解这样的事务设计,还需要从基本的Sql事务控制开始入手。虽然spring提供了更强大更灵活的事务控制方式,不过有些开发者也许更喜欢原生的事务控制方式,因为这用起来清晰明了。也不需要增加对Spring事务设计学习的成本。

原生SQL事务开启方式
SET [GLOBAL | SESSION] TRANSACTION
    transaction_characteristic [, transaction_characteristic] ...

transaction_characteristic:
    ISOLATION LEVEL level
  | READ WRITE
  | READ ONLY

level:
     REPEATABLE READ
   | READ COMMITTED
   | READ UNCOMMITTED
   | SERIALIZABLE

注意:MySQL本身是不支持嵌套事务的。在MySQL中开启一个事务后,如果在为提交前又开启了一个事务,则MySQL会自动提交当前事务。而Spring在实现嵌套事务时是通过SavePoint实现。SavePoint介绍:http://dev.mysql.com/doc/refman/5.7/en/savepoint.html

MySQL JDBC事务操作方式
Connection conn = DriverManager.getConnection(dbURL,dbUser,dbPassword);
conn.setAutoCommit(false);

try(Connection conn = DriverManager.getConnection(dbURL,dbUser,dbPassword);){
   conn.setAutoCommit(false);

   // perform operations such as insert, update, delete here
   // ..

   // if everything is OK, commit the transaction
   conn.commit();

} catch(SQLException e) {
   // in case of exception, rollback the transaction
   conn.rollback();
}
Spring事务操作:以DataSourceTransactionManage为例


    


// 编程中实际事务的使用方式
public void insert(User user) {
    DefaultTransactionDefinition def = new DefaultTransactionDefinition();
    // 设置事务的名字
    def.setName("UserService.insertUser");
    // 设置事务的传播属性
    def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
    TransactionStatus status = txManager.getTransaction(def);
    try {
        userDao.insertUser(user);
        txManager.commit(status);
    } catch (Exception ex) {
        txManager.rollback(status);
        throw new RuntimeException(ex);
    }
}

这种在数据源上控制事务的方式设计是在请求的当前线程范围内,从数据中获取一个connection对象,底层通过JDBC的方式操作当前connection的事务,是对原生JDBC的一种封装方式。所以这种事务的有效范围也只限于当前请求中的一个connection。在实际使用中,也许不会像上面这样通过编程的方式控制事务,一般会采用AOP或者加注解的方式,这种方式我们称之为声明式事务控制,原理也是一样,只是采用代理的方式帮我们省去了这些基本上都是重复的事务控制逻辑。

在Spring的事务实现中,PlatformTransactionManager是各种类型的事务的顶层抽象API,下面是Spring事务的类图

Spring事务传播属性配置
传播属性 描述 备注
required 如果当前没有事务就打开一个新的
requires new 新建事务,如果存在事务就把当前事务挂起 挂起对应的底层操作是:为当前事务新创建一个connection
support 如果当前有事务,就在事务中执行,如果没有就以非事务的方式执行
never 以非事务的方式运行,如果有事务则抛出异常
not supported 以非事务的方式运行,如果当前存在事务则挂起当前事务
nested 如果当前有事务,则在嵌套内执行,如果没有和required 一样 mysql是不支持嵌套事务的,它是通过MySQL提供的SavePoint功能实现
mandatory 使用当前的事务,如果没有就抛出异常

另外需要注意的地方:如果采用AOP或者注解这种方式,即声明式事务方式,在同一个类中,A方法调用B方法这种情况下,B方法的事务声明并不会生效:http://stackoverflow.com/questions/18590170/transactional-does-not-work-on-method-level

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/48635.html

相关文章

  • Spring【DAO模块】就是这么简单

    摘要:连接对象执行命令对象执行关闭值得注意的是,对数据库连接池是有很好的支持的。给我们提供了事务的管理器类,事务管理器类又分为两种,因为的事务和的事务是不一样的。 前言 上一篇Spring博文主要讲解了如何使用Spring来实现AOP编程,本博文主要讲解Spring的DAO模块对JDBC的支持,以及Spring对事务的控制... 对于JDBC而言,我们肯定不会陌生,我们在初学的时候肯定写过非...

    NSFish 评论0 收藏0
  • ORM “杀器”之 JOOQ

    摘要:摘要介绍简单实用,以及相对于传统框架的不同点。最令人满意的就是在实际使用过程中解决问题的灵活性。当前在数据服务组担任开发工程师,主要负责服务器开发。 摘要 介绍JOOQ简单实用,以及相对于传统ORM框架的不同点。 showImg(https://segmentfault.com/img/remote/1460000006763840); (图片来自http://www.jooq.org...

    GHOST_349178 评论0 收藏0
  • ORM “杀器”之 JOOQ

    摘要:摘要介绍简单实用,以及相对于传统框架的不同点。最令人满意的就是在实际使用过程中解决问题的灵活性。当前在数据服务组担任开发工程师,主要负责服务器开发。 摘要 介绍JOOQ简单实用,以及相对于传统ORM框架的不同点。 showImg(https://segmentfault.com/img/remote/1460000006763840); (图片来自http://www.jooq.org...

    Andrman 评论0 收藏0
  • ORM “杀器”之 JOOQ

    摘要:摘要介绍简单实用,以及相对于传统框架的不同点。最令人满意的就是在实际使用过程中解决问题的灵活性。当前在数据服务组担任开发工程师,主要负责服务器开发。 摘要 介绍JOOQ简单实用,以及相对于传统ORM框架的不同点。 showImg(https://segmentfault.com/img/remote/1460000006763840); (图片来自http://www.jooq.org...

    elarity 评论0 收藏0
  • ORM “杀器”之 JOOQ

    摘要:摘要介绍简单实用,以及相对于传统框架的不同点。最令人满意的就是在实际使用过程中解决问题的灵活性。当前在数据服务组担任开发工程师,主要负责服务器开发。 摘要 介绍JOOQ简单实用,以及相对于传统ORM框架的不同点。 showImg(/img/remote/1460000006763840); (图片来自http://www.jooq.org/) 正文 JOOQ是啥? JOOQ 是基于Ja...

    yeooo 评论0 收藏0

发表评论

0条评论

Euphoria

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<