Spring 5.x 源码之旅-64深入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下的验证结果当作最终结果的水货们请闭嘴】

 

传播机制图

AOP拦截原理

简单的示意图:


前面我们实战了7中传播机制,现在我们就来看下原理吧,源码比较复杂,我只选重点的讲。首先前面说了AOP事务的初始化,我们知道事务是基于AOP的拦截的,前面创建了JdkDynamicAopProxy


调用里面的invoke方法,这个在讲AOP的时候也讲过,其实就是获取拦截器链:

将目标对象的方法封装成MethodInvocation,然后执行拦截器,里面是个递归执行的方法:


最后执行的是拦截器的方法,也就是事务拦截器TransactionInterceptorinvoke

TransactionAspectSupport的invokeWithinTransaction

简单的一次事务就是这个样子,但是目标方法里面可能还有其他方法,也是有事务的,就涉及到事务的传播机制啦。


内部就是获取事务属性源TransactionAttributeSource,然后获取方法的事务注解属性,如果缓存里没有,就会去解析方法上的事务注解,然后返回注解属性,然后获取事务管理器,也就是我们注入的DataSourceTransactionManager,之后创建事务信息,这个最重要,是为了回滚和提交用的,最后调用目标方法。

    @Nullable
    	protected Object invokeWithinTransaction(Method method, @Nullable Class<?> targetClass,
    			final InvocationCallback invocation) throws Throwable {
    
    		// If the transaction attribute is null, the method is non-transactional.
    		TransactionAttributeSource tas = getTransactionAttributeSource();//事务注解属性
    		final TransactionAttribute txAttr = (tas != null ? tas.getTransactionAttribute(method, targetClass) : null);
    		final TransactionManager tm = determineTransactionManager(txAttr);//获取事务管理器
    
    		...
    
    		PlatformTransactionManager ptm = asPlatformTransactionManager(tm);
    		final String joinpointIdentification = methodIdentification(method, targetClass, txAttr);//连接点唯一标识 类名+方法名
    
    		if (txAttr == null || !(ptm instanceof CallbackPreferringPlatformTransactionManager)) {
    			// Standard transaction demarcation with getTransaction and commit/rollback calls.
    			TransactionInfo txInfo = createTransactionIfNecessary(ptm, txAttr, joinpointIdentification);
    
    			Object retVal;
    			try {
    			
    				retVal = invocation.proceedWithInvocation();//调用方法
    			}
    			catch (Throwable ex) {
    
    				completeTransactionAfterThrowing(txInfo, ex);
    				throw ex;
    			}
    			finally {
    				cleanupTransactionInfo(txInfo);//清除事务信息,恢复线程私有的老的事务信息
    			}
    
    			...
    			//成功后提交,会进行资源储量,连接释放,恢复挂起事务等操作
    			commitTransactionAfterReturning(txInfo);
    			return retVal;
    		}
    
    
    	}

注入的事务管理器:

TransactionAspectSupport的determineTransactionManager获取事务管理器

这个方法就是要获取事务管理器,如果前面没注入事务的话getBean会报异常:

TransactionAspectSupport的createTransactionIfNecessary创建事务信息(重点)

首先会生成一个代理事务属性,然后进行事务的获取,主要还是分析事务注解上的属性来进行事务状态的创建,最后创建事务信息,将事务状态放入事务信息中。


重点就在status = tm.getTransaction(txAttr);这句代码中,下一篇详细说吧。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值