Spring事务——使用TransactionProxyFactoryBean创建事务代理

Easter79
• 阅读 551

    Spring同时支持编程式事务策略和声明式事务策略,大部分时候,我们都推荐采用声明式事务策略。使用声明式事务策略的优势十分明显:

  • 声明式事务能大大降低开发者的代码书写量,而且声明式事务几乎不影响应用的代码。因此,不论底层事务策略如何变化,应用程序都无需任何改变

  • 应用程序代码无需任何事务处理代码,可以更专注于业务逻辑的实现

  • Spring可对任何POJO的方法提供事务管理,而且Spring的声明式事务管理无需容器的支持,可在任何环境下使用

  • EJB的CMT无法提供声明式回滚规则;而通过配置文件,Spring可指定事务在遇到特定异常时自动回滚。Spring不仅可在代码中使用setRollbackOnly回滚事务,也可在配置文件中配置回滚规则

  • 由于Spring采用AOP的方式管理事务,因此,可以在事务回滚动作中插入用户自己的动作,而不仅仅是执行系统默认的回滚

    本文主要介绍Spring中声明式事务管理的使用。

    在Spring1.X中,声明式事务使用TransactionProxyFactoryBean来配置事务代理Bean。正如它的类名所暗示的,它是一个专门为目标Bean生成事务代理的工厂Bean。既然TransactionProxyFactoryBean产生的是事务代理Bean,可见Spring的声明式事务策略是基于Spring AOP的。

每个TransactionProxyFactoryBean为一个目标Bean生成一个事务代理Bean,事务代理的方法改写了目标Bean的方法,就是在目标Bean的方法执行之前加入开始事务,在目标Bean的方法正常结束之前提交事务,如果遇到特定异常则回滚。

TransactionProxyFactoryBean创建事务代理时,需要了解当前事务所处的环境,该环境属性通过PlatformTransactionManager实例(其实现类的实例)传入,而相关事务规则则在该Bean定义中给出。下面是一个简单的持久化测试程序,该程序插入两条数据,这两条数据完全相同,将违反唯一键约束:

package com.abc.dao.impl;

public class NewsDaoImpl implements NewsDao {
    private DataSource dataSource;
    public void setDataSource(DataSrouce dataSource) {
        this.dataSource = dataSource;
    }
    public void insert(String title, String content) {
        JdbcTemplate template = new JdbcTemplate(dataSource);
        template.update("insert into news_table values (....)");
        //两次相同的操作,将违反主键约束
        template.update("insert into news_table values (....)");
    }
}

    上面的程序中,两次update语句将会违反主键约束——该行代码将会引发异常,如果在没有事务的环境下,前一条代码会向数据库中插入一条记录;但如果在增加了事务控制的环境下,则这两条语句是一个整体,因为第二条语句插入失败将导致第一条插入的记录也被回滚。下面是在Spring配置文件中配置该测试程序,并使用TransactionProxyFactoryBean为它们配置事务代理:

<!-- 使用C3P0数据库连接池作为数据源 -->
<bean id="dataSource" 
    class="com.mchange.v2.c3p0.ComboPolledDataSource" destroy-method="close">
    <property name="driverClass" value="com.mysql.jdbc.Driver" />
    <property name="jdbcUrl" value="jdbc:mysql://localhost/test" />
    <property name="user" value="root" />
    <property name="password" value="root" />
    <property name="maxPoolSize" value="40" />
    <property name="minPoolSize" value="4" />
    <property name="initialPoolSize" value="10" />
    <!-- 指定数据库连接池的连接的最大空闲时间 -->
    <property name="maxIdleTime" value="20" />
</bean>

<!-- 配置JDBC数据源的局部事务管理器,使用DataSourceTransactionManager类,该类实现了
PlatformTransactionManager接口,是针对采用数据源连接的特定实现 -->
<bean id="transactionManager" 
    class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <!-- 配置TransactionManager时需要注入数据源引用 -->
    <property name="dataSource" ref="dataSource" />
</bean>

<!-- 下面这个是前面定义的业务Bean -->
<bean id="newsDao" class="com.abc.dao.impl.NewsDaoImpl">
    <!-- 为业务Bean注入属性 -->
    <property name="dataSource" ref="dataSource" />
</bean>

<bean id="newsDaoTransProxy" 
    class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
    <!-- 为事务代理工厂Bean注入事务管理器 -->
    <property name="transactionManager" ref="transactionManager" />
    <!-- 要在哪个Bean上面创建事务代理对象 -->
    <property name="target" ref="newsDao" />
    <!-- 指定事务属性 -->
    <property name="transactionAttributes">
        <props>
            <prop key="*">PROPAGATION_REQUIRED</prop>
        </props>
    </property>
</bean>

    上面的配置文件中定义了一个事务管理器transactionManager,该事务管理器是针对JDBC局部事务的特定实现类。配置事务代理(如上面的newsDaoTransProxy)时需要传入一个事务管理器,一个目标Bean,并指定该事务代理的事务属性。事务属性由transactionAttributes属性指定。上面事务属性只有一条事务传播规则,该规则制定对于所有方法都使用PROPAGATION_REQUIRED的传播规则。Spring支持的事务传播规则如下:

  • PROPAGATION_MANDATORY:要求调用该方法的线程必须处于事务环境中,否则抛出异常

  • PROPAGATION_NESTED:如果执行该方法的线程已处于事务环境下,依然启动新的事务,方法在嵌套的事务里执行。如果执行方法的线程为处于事务中,也启动新的事务,然后执行该方法,此时与PROPAGATION_REQUIRED相同

  • PROPAGATION_NEVER:不允许调用该方法的线程处于事务环境下,如果调用该方法的线程处于事务环境下,则抛出异常

  • PROPAGATION_NOT_SUPPORTED:如果调用该方法的线程处在事务中,则暂停当前事务,然后执行该方法

  • PROPAGATION_REQUIRED:要求在事务环境中执行该方法,如果当前执行的线程已处于事务中,则直接调用;如果当前执行线程不处于事务中,则启动新的事务后执行该方法

  • PROPAGATION_REQUIRES_NEW:要求在事务环境中执行该方法,如果当前执行的线程已处于事务中,则暂停当前事务,启动新事务后执行该方法;如果当前执行线程不处于事务中,则启动新的事务后执行该方法

  • PROPAGATION_SUPPORTS:如果当前执行线程处于事务中,则使用当前事务;不过不在事务中,则不使用事务

    主程序中主要获取了定义的NewsDao类型的Bean,并调用其insert方法,下面是主程序:

public class SpringTest {
    public static void main(String[] arg) {
        ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
        NewsDao dao = (NewsDao)context.getBean("newsDaoTransProxy",NewsDao.class);
        dao.insert("新闻标题","新闻内容");
    }
}

    上面的第4行中获取了newsDaoTransProxy Bean,该Bean已经不在是NewsDaoImpl类的实例了,它只是Spring容器创建的事务代理,该事务代理以NewsDaoImpl实例为目标对象,且该目标对象也实现了NewsDao接口(与NewsDaoImpl实现了相同的接口),故代理对象也可以当成NewsDao实例来使用。运行上面的程序,将出现一个异常,而且insert方法所执行的两条SQL语句全部回滚——因为事务控制的缘故。

    当我们使用TransactionProxyFactoryBean为目标Bean配置了事务代理以后,SpringAOP将会把负责事务操作的增强处理织入目标Bean的业务方法当中。事实上,Spring不仅支持对接口的代理,整合CGLIB后,Spring甚至可以对具体类生成代理,只要设置proxyTargetClass属性为true即可。如果目标Bean没有实现任何接口,proxyTargetClass属性默认被设为true,此时Spring会对具体类生成代理。当然通常建议面向接口编程,而不要面向具体的实现类编程。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Easter79 Easter79
3年前
springboot+mybatis 使用事务
一、一些概念声明式的事务管理是基于AOP的,在springboot中可以通过@Transactional注解的方式获得支持,这种方式的优点是:1)非侵入式,业务逻辑不受事务管理代码的污染。2)方法级别的事务回滚,合理划分方法的粒度可以做到符合各种业务场景的事务管理。本文使用目前最常用的mybatis框架来配置springboot的事务
Wesley13 Wesley13
3年前
@Transactional 回滚不生效原因
事务的管理方式有两种,第一种是编程式事务管理,需要将数据库的自动提交等取消,并且需要自己编写事务代码,第二种则是声明式事务管理模式,spring利用springAOP特性编写了注解即题目中所提到的方式来管理事务,避免开发人员编写大量的事务代码。一、特性先来了解一下@Transactional注解的特性吧,可以更好排查问题1\.service类
Wesley13 Wesley13
3年前
spring事物管理
Spring提供了一流的事务管理。在Spring中可以支持声明式事务和编程式事务spring事物的概念和重要性事物的属性事物管理器声明事物1.1定义:事务管理在应用程序中起着至关重要的作用:它是一系列任务的组成工作单元,在这个工作单元中,所有的任务必须同时执行。它们只有二种可能执行结果,要么所有任务全
Easter79 Easter79
3年前
Spring学习笔记十六
Spring事务ACID原子性一致性隔离性持久性编程式/声明式Spring事务管理抽象是PlatformTransactionManagerJDBC:DataSourceTransactionManagerJTA:JtaTransactionManagerHibernate:Hibern
Wesley13 Wesley13
3年前
Spring声明式事务注解@Transactional
spring支持编程式事务管理和声明式事务管理两种方式。编程式事务管理使用TransactionTemplate或者直接使用底层的PlatformTransactionManager。对于编程式事务管理,spring推荐使用TransactionTemplate。声明式事务管理建立在AOP之上的。其本质是对方法前后进行拦截,然后在目标方法开始之前创
Wesley13 Wesley13
3年前
Spring事务管理
Spring提供了一流的事务管理。在Spring中可以支持声明式事务和编程式事务。    本章主要目标如下:     1,Spring事务    2,事务属性    3,事务管理器    4,声明式事务      1.1Spring的事务     事务管理在应用程序中起着至关重要的作用:它是一系列任务
线上SQL超时场景分析-MySQL超时之间隙锁 | 京东物流技术团队
前言之前遇到过一个由MySQL间隙锁引发线上sql执行超时的场景,记录一下。背景说明分布式事务消息表:业务上使用消息表的方式,依赖本地事务,实现了一套分布式事务方案消息表名:mqmessages数据量:3000多万索引:createtime和statuss
京东云开发者 京东云开发者
9个月前
Spring事务实现原理
1、引言spring的springtx模块提供了对事务管理支持,使用spring事务可以让我们从复杂的事务处理中得到解脱,无需要去处理获得连接、关闭连接、事务提交和回滚等这些操作。spring事务有编程式事务和声明式事务两种实现方式。编程式事务是通过编写代
京东云开发者 京东云开发者
1星期前
Spring事务实现原理
作者:京东零售范锡军1、引言spring的springtx模块提供了对事务管理支持,使用spring事务可以让我们从复杂的事务处理中得到解脱,无需要去处理获得连接、关闭连接、事务提交和回滚等这些操作。spring事务有编程式事务和声明式事务两种实现方式。编
Easter79
Easter79
Lv1
今生可爱与温柔,每一样都不能少。
文章
2.8k
粉丝
5
获赞
1.2k