Spring 5.x 源码之旅-66深入AOP事务原理三

作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO

联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬

学习必须往深处挖,挖的越深,基础越扎实!

阶段1、深入多线程

阶段2、深入多线程设计模式

阶段3、深入juc源码解析


阶段4、深入jdk其余源码解析


阶段5、深入jvm源码解析

码哥源码部分

码哥讲源码-原理源码篇【2024年最新大厂关于线程池使用的场景题】

码哥讲源码【炸雷啦!炸雷啦!黄光头他终于跑路啦!】

码哥讲源码-【jvm课程前置知识及c/c++调试环境搭建】

​​​​​​码哥讲源码-原理源码篇【揭秘join方法的唤醒本质上决定于jvm的底层析构函数】

码哥源码-原理源码篇【Doug Lea为什么要将成员变量赋值给局部变量后再操作?】

码哥讲源码【你水不是你的错,但是你胡说八道就是你不对了!】

码哥讲源码【谁再说Spring不支持多线程事务,你给我抽他!】

终结B站没人能讲清楚红黑树的历史,不服等你来踢馆!

打脸系列【020-3小时讲解MESI协议和volatile之间的关系,那些将x86下的验证结果当作最终结果的水货们请闭嘴】

创建事务信息流程图

传播机制图

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来获取连接。

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

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值