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

传播机制图

202309162319139741.png

AOP拦截原理

简单的示意图:

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

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

202309162319155504.png

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

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

202309162319163716.png

202309162319168127.png

TransactionAspectSupport的invokeWithinTransaction

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

202309162319172518.png
内部就是获取事务属性源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;
    		}
    
    
    	}

注入的事务管理器:

202309162319176829.png

TransactionAspectSupport的determineTransactionManager获取事务管理器

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

2023091623191822410.png

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

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

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

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


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] ,回复【面试题】 即可免费领取。

阅读全文