首页 > 其他分享 >Spring进阶篇(7)-TransactionSynchronizationManager(事务监听)

Spring进阶篇(7)-TransactionSynchronizationManager(事务监听)

时间:2024-04-03 10:45:33浏览次数:22  
标签:TransactionSynchronizationManager 事务 transaction Spring void ThreadLocal 进阶篇 Con

转载自:https://www.jianshu.com/p/4b5eb29cc6d9

JAVA && Spring && SpringBoot2.x — 学习目录

TransactionSynchronizationManager是事务同步管理器。我们可以自定义实现TransactionSynchronization类,来监听Spring的事务操作。可以在事务提交之后,回调TransactionSynchronization类的方法。

1. TransactionSynchronizationManager在源码中的使用

在SpringCache的自定义CacheManager中。装饰Cache对象使其支持事务操作。即只有在事务提交成功之后,才会进行缓存。

源码位置:org.springframework.cache.transaction.TransactionAwareCacheDecorator#put

    @Override
    public void put(final Object key, @Nullable final Object value) {
        if (TransactionSynchronizationManager.isSynchronizationActive()) {
            TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronizationAdapter() {
                @Override
                public void afterCommit() {
                    TransactionAwareCacheDecorator.this.targetCache.put(key, value);
                }
            });
        }
        else {
            this.targetCache.put(key, value);
        }
    }

【JDBC中的connection详解】中,我们知道connection是线程不安全的,即需要为每一个数据库操作都获取一个Connection对象。事务操作可以看做是一个整体,必须使用同一个Connection进行操作。故在Spring中使用LocalThread(线程上下文)将Connection对象和线程绑定。

在Spring中的org.springframework.transaction.support.TransactionSynchronizationManager类中,便是使用ThreadLocal来为不同的事务线程提供独立的资源副本,并且同时维护这些事务的配置属性和运行状态。

2. Connection与TransactionSynchronizationManager关系

1. 请求事务方法时,调用dobegin()将事务信息保存到TransactionSynchronizationManager中:在该方法中主要流程是在数据库连接池中获取一个Connection对象,然后将Connection对象放入到ThreadLocal中。实际上该事务方法的信息均由TransactionSynchronizationManager类管理。

源码:org.springframework.jdbc.datasource.DataSourceTransactionManager#doBegin

   protected void doBegin(Object transaction, TransactionDefinition definition) {
        DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;
        Connection con = null;

        try {
            if (!txObject.hasConnectionHolder() ||
                    txObject.getConnectionHolder().isSynchronizedWithTransaction()) {
                    //在可见的数据源(连接池)中获取Connection对象
                Connection newCon = obtainDataSource().getConnection();
                if (logger.isDebugEnabled()) {
                    logger.debug("Acquired Connection [" + newCon + "] for JDBC transaction");
                }
                        
                txObject.setConnectionHolder(new ConnectionHolder(newCon), true);
            }
            
            ...
          //关闭Connection对象的自动提交
           if (con.getAutoCommit()) {
                txObject.setMustRestoreAutoCommit(true);
                if (logger.isDebugEnabled()) {
                    logger.debug("Switching JDBC Connection [" + con + "] to manual commit");
                }
                con.setAutoCommit(false);
            }

            //将Connection对象绑定到Thread中
            if (txObject.isNewConnectionHolder()) {
                TransactionSynchronizationManager.bindResource(obtainDataSource(), txObject.getConnectionHolder());
            }
        }

        catch (Throwable ex) {
            if (txObject.isNewConnectionHolder()) {
                DataSourceUtils.releaseConnection(con, obtainDataSource());
                txObject.setConnectionHolder(null, false);
            }
            throw new CannotCreateTransactionException("Could not open JDBC Connection for transaction", ex);
        }
    }
public abstract class TransactionSynchronizationManager {
    private static final ThreadLocal<Map<Object, Object>> resources =
            new NamedThreadLocal<>("Transactional resources");

    public static void bindResource(Object key, Object value) throws IllegalStateException {
        Object actualKey = TransactionSynchronizationUtils.unwrapResourceIfNecessary(key);
        Assert.notNull(value, "Value must not be null");
       
        Map<Object, Object> map = resources.get();
        // set ThreadLocal Map if none found
        if (map == null) {
            map = new HashMap<>();
            resources.set(map);
        }
        //将Connection对象绑定到resources 上。
        Object oldValue = map.put(actualKey, value);
        ...
    }
}
  1. 执行sql语句时,实际上通过org.mybatis.spring.transaction.SpringManagedTransaction类直接获取Connection对象。

源码:org.mybatis.spring.transaction.SpringManagedTransaction#openConnection

  private void openConnection() throws SQLException {
    //在ThreadLocal中获取Connection对象
    this.connection = DataSourceUtils.getConnection(this.dataSource);
    this.autoCommit = this.connection.getAutoCommit();
    //在ThreadLocal中获取是否开启事务
    this.isConnectionTransactional = DataSourceUtils.isConnectionTransactional(this.connection, this.dataSource);
   ...
  }

源码:org.springframework.jdbc.datasource.DataSourceUtils#doGetConnection

    public static Connection doGetConnection(DataSource dataSource) throws SQLException {
        Assert.notNull(dataSource, "No DataSource specified");
        //在doBegin()方法中,已经将创建的Connection对象放入到TransactionSynchronizationManager中
        ConnectionHolder conHolder = (ConnectionHolder) TransactionSynchronizationManager.getResource(dataSource);
        if (conHolder != null && (conHolder.hasConnection() || conHolder.isSynchronizedWithTransaction())) {
            conHolder.requested();
            if (!conHolder.hasConnection()) {
                logger.debug("Fetching resumed JDBC Connection from DataSource");
                conHolder.setConnection(fetchConnection(dataSource));
            }
            //直接返回Thread存储的Connection对象。
            return conHolder.getConnection();
        }

        ...

        return con;
    }

3. TransactionSynchronizationManager的结构


public abstract class TransactionSynchronizationManager {

     //线程上下文中保存着【线程池对象:ConnectionHolder】的Map对象。线程可以通过该属性获取到同一个Connection对象。
    private static final ThreadLocal<Map<Object, Object>> resources = new NamedThreadLocal<>("Transactional resources");

    //事务同步器,是Spring交由程序员进行扩展的代码,每个线程可以注册N个事务同步器。
    private static final ThreadLocal<Set<TransactionSynchronization>> synchronizations = new NamedThreadLocal<>("Transaction synchronizations");
    
    // 事务的名称  
    private static final ThreadLocal<String> currentTransactionName = new NamedThreadLocal<>("Current transaction name");
    // 事务是否是只读  
    private static final ThreadLocal<Boolean> currentTransactionReadOnly = new NamedThreadLocal<>("Current transaction read-only status");
    // 事务的隔离级别
    private static final ThreadLocal<Integer> currentTransactionIsolationLevel = new NamedThreadLocal<>("Current transaction isolation level");
    // 事务是否开启   actual:真实的
    private static final ThreadLocal<Boolean> actualTransactionActive = new NamedThreadLocal<>("Actual transaction active");
}

org.springframework.transaction.interceptor.TransactionInterceptor#invoke中,对事务方法进行拦截处理。在createTransactionIfNecessary创建TransactionInfo对象时,会调用AbstractPlatformTransactionManager#prepareSynchronization方法初始化事务同步器。

    protected void prepareSynchronization(DefaultTransactionStatus status, TransactionDefinition definition) {
        if (status.isNewSynchronization()) {
            TransactionSynchronizationManager.setActualTransactionActive(status.hasTransaction());
            TransactionSynchronizationManager.setCurrentTransactionIsolationLevel(
                    definition.getIsolationLevel() != TransactionDefinition.ISOLATION_DEFAULT ?
                            definition.getIsolationLevel() : null);
            TransactionSynchronizationManager.setCurrentTransactionReadOnly(definition.isReadOnly());
            TransactionSynchronizationManager.setCurrentTransactionName(definition.getName());
            //初始化事务同步器
            TransactionSynchronizationManager.initSynchronization();
        }
    }

4. TransactionSynchronization

这个类是程序员对事务同步的扩展点:用于事务同步回调的接口。

public interface TransactionSynchronization extends Flushable {

    /** Completion status in case of proper commit. */
    int STATUS_COMMITTED = 0;

    /** Completion status in case of proper rollback. */
    int STATUS_ROLLED_BACK = 1;

    /** Completion status in case of heuristic mixed completion or system errors. */
    int STATUS_UNKNOWN = 2;


    /**
     * 事务挂起
     * Supposed to unbind resources from TransactionSynchronizationManager if managing any.
     * @see TransactionSynchronizationManager#unbindResource
     */
    default void suspend() {
    }

    /**
     * 事务恢复
     * Supposed to rebind resources to TransactionSynchronizationManager if managing any.
     * @see TransactionSynchronizationManager#bindResource
     */
    default void resume() {
    }

    /**
     * 将基础会话刷新到数据存储区(如果适用),比如Hibernate/JPA的Session
     * @see org.springframework.transaction.TransactionStatus#flush()
     */
    @Override
    default void flush() {
    }

    /**
     * 在事务提交前触发,此处若发生异常,会导致回滚。
     * @see #beforeCompletion
     */
    default void beforeCommit(boolean readOnly) {
    }

    /**
     * 在beforeCommit之后,commit/rollback之前执行。即使异常,也不会回滚。
     * @see #beforeCommit
     * @see #afterCompletion
     */
    default void beforeCompletion() {
    }

    /**
     * 事务提交后执行。
     */
    default void afterCommit() {
    }

    /**
     * 事务提交/回滚执行
     */
    default void afterCompletion(int status) {
    }

一般而言,我们在TransactionSynchronization使用最多的是afterCommitafterCompletion方法。可以在事务执行完毕之后,直接调用afterCommit()方法进行异步通知。

我们在doCommit()方法中提交事务后,在cleanupAfterCompletion对connection进行重置,即我们依旧可以在afterCommit()回调中对数据库进行操作。

    private void processCommit(DefaultTransactionStatus status) throws TransactionException {
        try {
            //提交事务
            doCommit(status);
            ...
            try {
             //回调所有事务同步器的afterCommit方法。
                triggerAfterCommit(status);
            }
            finally {
            //回调所有事务同步器的afterCompletion方法。
                triggerAfterCompletion(status, TransactionSynchronization.STATUS_COMMITTED);
            }

        }
        finally {
           //清除TransactionSynchronizationManager的ThreadLocal绑定的数据。
           //解除Thread绑定的resources资源。
           //将Commit设置为自动提交。
          //清理ConnectionHolder资源。
            cleanupAfterCompletion(status);
        }
    }

文章参考

如何在数据库事务提交成功后进行异步操作

自定义springcache实现事务提交后处理缓存

Spring事务监听机制---使用@TransactionalEventListener处理数据库事务提交成功后再执行操作(附:Spring4.2新特性讲解)

标签:TransactionSynchronizationManager,事务,transaction,Spring,void,ThreadLocal,进阶篇,Con
From: https://www.cnblogs.com/wanghengbin/p/18112147

相关文章

  • 面试类 - Spring基础(一)
    1.Spring是什么?特性?有哪些模块?                                 SpringLogo一句话概括:Spring是一个轻量级、非入侵式的控制反转(IoC)和面向切面(AOP)的框架。2003年,一个音乐家RodJohnson决定发......
  • springboot中的Lombok的使用
    1.Lombok       属于一种对实体类进行简化配置的功能操作,通过@Data实现实体类中get和set方法省略行为。2.测试方式  第一步:创建一个springboot的项目,并同时选择需要的依赖         第二步:创建bean实体类,同时使用@Data注解       第三步:测试,此......
  • springboot的自动装配
    SpringBoot自动装配       Spring框架提供了IOC的功能实现对所有的javabean进行装配,当时使用的xml文件       提供bean配置完成注入行为。       在SpringBoot中可以使用自动装配配置类替换之前xml文件方式。       使用@Configuration修饰一个类......
  • springboot234基于Spring Boot的疗养院管理系统的设计与实现
    具体演示视频链接:https://pan.baidu.com/s/1epOAnmfyRpfuI3eBR13WDA?pwd=8888毕业设计(论文)疗养院管理系统设计与实现摘要传统办法管理信息首先需要花费的时间比较多,其次数据出错率比较高,而且对错误的数据进行更改也比较困难,最后,检索数据费事费力。因此,在计算机上安......
  • springboot235基于SpringBoot的房屋交易平台的设计与实现
    具体演示视频链接:https://pan.baidu.com/s/1epOAnmfyRpfuI3eBR13WDA?pwd=8888本科毕业设计论文题目:房屋交易平台设计与实现系别:XX系(全称)专业:软件工程班级:软件工程15201学生姓名:学生学号:指导教师:导师1导师2摘要信息数据从传统到当代,是一直在变革当中,突......
  • springboot的常用标签
    SpringBoot中常用的注解       -------       类注解:       @RequestMapping  --方法映射       @ResponseBody    --JSON返回修饰       @RestController  --控制器JSON返回控制       ---------------      ......
  • 12.Springsecurity简单总结
    关于springsecurity的介绍后面我接触的应该是这个和Shiro!!!一个网站很重要很重要的是安全问题(狂神说的)哈哈我觉得更重要的是编写吧来看吧maven依赖这个肯定很重要thymeleaf依赖就跳过了这个东西应该很重要我学到现在一直离不开当然我还是没完全搞懂语法嘞就像jsp......
  • 【附源码】JAVA计算机毕业设计智慧点餐系统(springboot+mysql+开题+论文)
    本系统(程序+源码)带文档lw万字以上 文末可获取一份本项目的java源码和数据库参考。系统程序文件列表开题报告内容研究背景随着信息技术的快速发展和互联网的普及,人们的生活方式发生了深刻的变化。特别是在餐饮行业,传统的点餐方式已经无法满足现代消费者对于便捷性、个性化......
  • Spring Data Elasticsearch String类型不指定filed 索引创建情况
    对于String类型的字段如果不指定类型会默认创建两种倒排索引       "itemSkuCodes":{         "type":"text",         "fields":{           "keyword":{             "type":"keyword",           ......
  • 基于SpringBoot+Vue前后端分离的宠物领养管理系统的设计与实现+15000字毕业论文
    介绍用户端:首页:播放宠物视频,展示公告列表,介绍流浪宠物。宠物领养:用户搜索想要领养宠物,申请领养,查看自己领养的宠物。流浪宠物救助:用户能够看到需要救助的流浪宠物,并能够新增新的流浪宠物信息。宠物喂养点:用户能够看到需要喂养的流浪宠物的地点,并展示出地点环境。丢失宠物......