2023-09-16  阅读(1)
原文作者:王伟王胖胖 原文地址: https://blog.csdn.net/wangwei19871103/article/details/105344469

创建事务信息流程图

202309162319251801.png

传播机制图

202309162319260262.png

AbstractPlatformTransactionManager的suspend挂起当前事务

有些传播机制需要挂起当前事务,比如NOT_SUPPORTEDREQUIRES_NEW。首先会清除所有线程相关的同步状态,如果当前事务存在的话,就进行一些属性的清除,比如清空连接持有器,清空线程私有变量的同步状态,最后把当前事务清除的属性保存到一个SuspendedResourcesHolder里,以便于恢复的时候设置会去。

    @Nullable
    	protected final SuspendedResourcesHolder suspend(@Nullable Object transaction) throws TransactionException {
    		if (TransactionSynchronizationManager.isSynchronizationActive()) {
    			List<TransactionSynchronization> suspendedSynchronizations = doSuspendSynchronization();
    			try {
    				Object suspendedResources = null;
    				if (transaction != null) {
    					suspendedResources = doSuspend(transaction);//挂起的资源,连接持有器
    				}
    				String name = TransactionSynchronizationManager.getCurrentTransactionName();//当前事务名字
    				TransactionSynchronizationManager.setCurrentTransactionName(null);//取消绑定
    				boolean readOnly = TransactionSynchronizationManager.isCurrentTransactionReadOnly();//当前事务可读性
    				TransactionSynchronizationManager.setCurrentTransactionReadOnly(false);
    				Integer isolationLevel = TransactionSynchronizationManager.getCurrentTransactionIsolationLevel();//当前事务隔离级别
    				TransactionSynchronizationManager.setCurrentTransactionIsolationLevel(null);
    				boolean wasActive = TransactionSynchronizationManager.isActualTransactionActive();//当前事务激活状态
    				TransactionSynchronizationManager.setActualTransactionActive(false);
    				return new SuspendedResourcesHolder(
    						suspendedResources, suspendedSynchronizations, name, readOnly, isolationLevel, wasActive);
    			}
    			catch (RuntimeException | Error ex) {
    				// doSuspend failed - original transaction is still active...
    				doResumeSynchronization(suspendedSynchronizations);
    				throw ex;
    			}
    		}
    		else if (transaction != null) {
    			// Transaction active but no synchronization active.
    			Object suspendedResources = doSuspend(transaction);
    			return new SuspendedResourcesHolder(suspendedResources);
    		}
    		else {
    			// Neither transaction nor synchronization active.
    			return null;
    		}

DataSourceTransactionManager的doSuspend挂起资源

    	@Override
    	protected Object doSuspend(Object transaction) {
    		DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;
    		txObject.setConnectionHolder(null);//清空连接持有器
    		return TransactionSynchronizationManager.unbindResource(obtainDataSource());//解绑线程私有的资源
    	}

AbstractPlatformTransactionManager的SuspendedResourcesHolder构造方法

其实就是保存了挂起事务的信息啦。

    private SuspendedResourcesHolder(
    				@Nullable Object suspendedResources, List<TransactionSynchronization> suspendedSynchronizations,
    				@Nullable String name, boolean readOnly, @Nullable Integer isolationLevel, boolean wasActive) {
    
    			this.suspendedResources = suspendedResources;//连接持有器
    			this.suspendedSynchronizations = suspendedSynchronizations;//同步状态
    			this.name = name;//方法名
    			this.readOnly = readOnly;//是否只读
    			this.isolationLevel = isolationLevel;//隔离级别,默认mysql是可重复度,oracle是提交读
    			this.wasActive = wasActive;//事务是否激活
    		}

AbstractPlatformTransactionManager的newTransactionStatus创建事务状态

这里有个参数很重要newTransaction,是否是新连接,比如当前事务不存在的情况下,肯定是true,但是如果存在,就有可能false,具体还是看传播机制。

    	protected DefaultTransactionStatus newTransactionStatus(
    			TransactionDefinition definition, @Nullable Object transaction, boolean newTransaction,
    			boolean newSynchronization, boolean debug, @Nullable Object suspendedResources) {
    		//是否要新同步,只有要新同步且当前无同步激活事务
    		boolean actualNewSynchronization = newSynchronization &&
    				!TransactionSynchronizationManager.isSynchronizationActive();
    		return new DefaultTransactionStatus(
    				transaction, newTransaction, actualNewSynchronization,
    				definition.isReadOnly(), debug, suspendedResources);
    	}

DefaultTransactionStatus创建默认事务状态

其实就是做个记录,这次事务的状态。

    	public DefaultTransactionStatus(
    			@Nullable Object transaction, boolean newTransaction, boolean newSynchronization,
    			boolean readOnly, boolean debug, @Nullable Object suspendedResources) {
    
    		this.transaction = transaction;//新创建事务
    		this.newTransaction = newTransaction;//是否需要新事务
    		this.newSynchronization = newSynchronization;//是否要新同步
    		this.readOnly = readOnly;//是否只读
    		this.debug = debug;//是否要debug
    		this.suspendedResources = suspendedResources;//是否有挂起的连接资源
    	}

DataSourceTransactionManager的doBegin开启连接和事务

这里就是开启新连接的地方,如果当前事务没有连接资源了,就会去创建一个新的连接,然后设置连接属性,做一些事务的标记等,表示是一个新的事务了。

    @Override
    	protected void doBegin(Object transaction, TransactionDefinition definition) {
    		DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;
    		Connection con = null;
    
    		try {
    			if (!txObject.hasConnectionHolder() ||//当前事务没有连接资源
    					txObject.getConnectionHolder().isSynchronizedWithTransaction()) {
    				Connection newCon = obtainDataSource().getConnection();//创建新连接
    				if (logger.isDebugEnabled()) {
    					logger.debug("Acquired Connection [" + newCon + "] for JDBC transaction");
    				}
    				txObject.setConnectionHolder(new ConnectionHolder(newCon), true);//设置连接持有器
    			}
    
    			txObject.getConnectionHolder().setSynchronizedWithTransaction(true);//标记事务同步状态
    			con = txObject.getConnectionHolder().getConnection();
    			//获取先前隔离级别,默认就是用数据库默认的
    			Integer previousIsolationLevel = DataSourceUtils.prepareConnectionForTransaction(con, definition);
    			txObject.setPreviousIsolationLevel(previousIsolationLevel);//设置先前隔离级别
    			txObject.setReadOnly(definition.isReadOnly());//设置是否只读
    
    			if (con.getAutoCommit()) {
    				txObject.setMustRestoreAutoCommit(true);//设置需要回复自动提交
    				if (logger.isDebugEnabled()) {
    					logger.debug("Switching JDBC Connection [" + con + "] to manual commit");
    				}
    				con.setAutoCommit(false);//关闭自动提交
    			}
    			//是否需要设置只读命令
    			prepareTransactionalConnection(con, definition);
    			txObject.getConnectionHolder().setTransactionActive(true);//标记激活事务
    
    			int timeout = determineTimeout(definition);
    			if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
    				txObject.getConnectionHolder().setTimeoutInSeconds(timeout);
    			}
    
    		
    			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);
    		}
    	}

AbstractDriverBasedDataSource的getConnection获取一个连接

其实就是设置用户名和密码,最终是调用JDBC来获取连接。

202309162319264563.png

202309162319268144.png

202309162319273025.png

202309162319281266.png

TransactionSynchronizationManager的bindResource绑定数据源和连接资源

这里就是数据源和连接资源的绑定。

202309162319284657.png

202309162319289368.png

好了,今天就到这里了,希望对学习理解有帮助,大神看见勿喷,仅为自己的学习理解,能力有限,请多包涵。


Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文