Transaction註解原理

專注的阿熊發表於2020-12-09

開啟事務註解 EnableTransactionManagement ,該註解往容器中匯入了匯入 TransactionManagementConfigurationSelector 元件。該元件有個方法,在容器重新整理的時候會被呼叫。 ( 此處不講解為什麼會被呼叫,重點講解 Transaction 註解 )

protected  String[] selectImports(AdviceMode adviceMode) {

switch (adviceMode) {

case PROXY:

return new String[] {AutoProxyRegistrar.class.getName(),

ProxyTransactionManagementConfiguration.class.getName()};

case ASPECTJ:

return new String[] {

TransactionManagementConfigUtils.TRANSACTION_ASPECT_CONFIGURATION_CLASS_NAME};

·  default:

·  return null;

·  }

·  }

· 

·  EnableTransactionManagement 預設是 **AdviceMode mode() default AdviceMode.PROXY;** 所以
AutoProxyRegistrar ProxyTransactionManagementConfiguration 會被載入到容器中。

· 

·  AutoProxyRegistrar 功能是往容器中註冊了 InfrastructureAdvisorAutoProxyCreator 是個 BeanPostProcessor 後置處理器。

·  ProxyTransactionManagementConfiguration 利用 @Bean 註解往容器中新增多個 bean (BeanFactoryTransactionAttributeSourceAdvisor 是個 Advisor 和前面 Aop @Before 一樣都是通知方法,但是此處的是手動匯入的, Aop 是自己自動生成的 )

· 

·  bean 生成完成以後會跟單網gendan5.com呼叫後置處理器初始化。會去查詢 bean 有沒有合適的通知方法。所以重點是在找通知方法,找到了合適的就會生成代理物件。

· 

·  // 簡單的分析一下流程

·  AbstractAutowireCapableBeanFactory#initializeBean() ->

·  AbstractAutowireCapableBeanFactory#applyBeanPostProcessorsAfterInitialization() ->

·  AbstractAutoProxyCreator#postProcessAfterInitialization() ->

·  AbstractAutoProxyCreator#wrapIfNecessary()->

·  AbstractAdvisorAutoProxyCreator#getAdvicesAndAdvisorsForBean()

/**

·   * 尋找可用的通知方法

·   */

·  protected List<Advisor> findEligibleAdvisors(Class<?> beanClass, String beanName) {

·  // 尋找所有的 Advisors, 會去容器遍歷所有型別為 Advisor bean

·  //BeanFactoryTransactionAttributeSourceAdvisor 就是我們手動匯入的 Advisor           

·  List<Advisor> candidateAdvisors = findCandidateAdvisors();

·  // 在所有的 Advisors, 看是否有 beanClass 匹配的

·  List<Advisor> eligibleAdvisors = findAdvisorsThatCanApply(candidateAdvisors, beanClass, beanName);

·  extendAdvisors(eligibleAdvisors);

·  if (!eligibleAdvisors.isEmpty()) {

·  eligibleAdvisors = sortAdvisors(eligibleAdvisors);

·  }

·  return eligibleAdvisors;

·  }

·  // 找到所有的 Advisors, 剩下就是是否有匹配的 findAdvisorsThatCanApply() 方法 // 最終走到 TransactionAttributeSourcePointcut#matches//->AbstractFallbackTransactionAttributeSource#computeTransactionAttribute

· 

·  ```javaprotected TransactionAttribute computeTransactionAttribute(Method method, Class<?> targetClass) {

·  // 判斷方法修飾符是不是 Public allowPublicMethodsOnly() 預設是 true

·  if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) {

·  return null;

·  }

· 

·  // 這裡面就是判斷方法上有沒有 Transactional 註解,有進行解析操作並返回。

·  TransactionAttribute txAttr = findTransactionAttribute(specificMethod);

·  if (txAttr != null) {

·  return txAttr;

·  }

·  // 程式碼每貼全,為了少點沒用的東西。。。。

·  那要是找了合適的通知方法,就生成代理物件,並設定回撥函式。設定回撥函式是CglibAopProxy.DynamicAdvisedInterceptor intercept 方法,該方法有個 getInterceptorsAndDynamicInterceptionAdvice 獲取合適的攔截器。 ( 事務的攔截器是 TransactionInterceptor)

·  public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {

·                  // 獲取目標物件,不是代理物件

·                  target = getTarget();

·  if (target != null) {

·  targetClass = target.getClass();

·  }

·  // 獲取方法合適的攔截器

·  List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass);

·  Object retVal;

·  // 沒有合適的攔截器並且方法的修飾符是 Public

·  if (chain.isEmpty() && Modifier.isPublic(method.getModifiers())) {

·  Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args);

·  // 會呼叫目標方法

·  retVal = methodProxy.invoke(target, argsToUse);

·  }

·  else {

·  /*

·  * 1. 有合適的攔截器 : 會去執行攔截器 TransactionInterceptor#invoke()

·                      *    事務的提交,回滾等一系列操作都在裡面。具體是調去了

·                      *    TransactionAspectSupport#invokeWithinTransaction() 方法   

·  *  2. 沒有合適的攔截器並且方法的修飾符不是 Public: 這裡面也會去調目標方法

·  *     ( 這裡簡單可以走進去看一下 )

·  */

·  retVal = new CglibMethodInvocation(proxy, target, method, args, targetClass, chain, methodProxy).proceed();

·  }

·  retVal = processReturnType(proxy, target, method, retVal);

·  return retVal;

·  }


來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69946337/viewspace-2740620/,如需轉載,請註明出處,否則將追究法律責任。

相關文章