欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

spring是如何實現(xiàn)聲明式事務(wù)的

 更新時間:2021年04月20日 08:27:18   作者:是時候改個好點的名字  
這篇文章主要介紹了spring是如何實現(xiàn)聲明式事務(wù)的,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

前言

今天我們來講講spring的聲明式事務(wù)。

開始

說到聲明式事務(wù),我們現(xiàn)在回顧一下事務(wù)這個概念,什么是事務(wù)呢,事務(wù)指的是邏輯上的⼀組操作,組成這組操作的各個單元,要么全部成功,要么全部不成功。從而確保了數(shù)據(jù)的準確與安全。事務(wù)有著四大特性(ACID),分別是
原子性(Atomicity)原⼦性是指事務(wù)是⼀個不可分割的⼯作單位,事務(wù)中的操作要么都發(fā)⽣,要么都不發(fā)⽣。

⼀致性(Consistency)事務(wù)必須使數(shù)據(jù)庫從⼀個⼀致性狀態(tài)變換到另外⼀個⼀致性狀態(tài)。

隔離性(Isolation)事務(wù)的隔離性是多個⽤戶并發(fā)訪問數(shù)據(jù)庫時,數(shù)據(jù)庫為每⼀個⽤戶開啟的事務(wù),每個事務(wù)不能被其他事務(wù)的操作數(shù)據(jù)所⼲擾,多個并發(fā)事務(wù)之間要相互隔離。

持久性(Durability) 持久性是指⼀個事務(wù)⼀旦被提交,它對數(shù)據(jù)庫中數(shù)據(jù)的改變就是永久性的,接下來即使數(shù)據(jù)庫發(fā)⽣故障

也不應(yīng)該對其有任何影響。

在spring中,一共有兩種方式可以實現(xiàn)事務(wù)控制,分別是編程式事務(wù)和聲明式事務(wù)。編程式事務(wù)指的是在代碼中添加事務(wù)控制代碼,而聲明式事務(wù)指的是利用xml或者注解的形式來配置控制事務(wù),下面就以純注解配置聲明式事務(wù)為例進行剖析。

spring開啟聲明式事務(wù)的注解是@EnableTransactionManagement,講到這里首先要明白一點,spring的事務(wù)管理器管理事務(wù)其實就是利用aop的方式,通過創(chuàng)建動態(tài)代理加上攔截,實現(xiàn)的事務(wù)管理。在spring的配置類中加上這個注解,就支持了聲明式事務(wù),那么spring是怎么通過這么一個注解就可以支持事務(wù)的呢,我們來看代碼。
首先我們看到,在這個注解上,import了一個selector

@Import(TransactionManagementConfigurationSelector.class)

我們看這個selector類中的這么一段代碼

@Override
 protected String[] selectImports(AdviceMode adviceMode) {
  switch (adviceMode) {
   case PROXY:
    return new String[] {AutoProxyRegistrar.class.getName(),
      ProxyTransactionManagementConfiguration.class.getName()};
   case ASPECTJ:
    return new String[] {determineTransactionAspectClass()};
   default:
    return null;
  }
 }

這段代碼中,引入了AutoProxyRegistrar和ProxyTransactionManagementConfiguration這兩個類,我們先來看AutoProxyRegistrar這個類,這個類中有一段這樣的代碼

if (mode == AdviceMode.PROXY) {
    //重要的是這句代碼
 AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry);
 if ((Boolean) proxyTargetClass) {
  AopConfigUtils.forceAutoProxyCreatorToUseClassProxying(registry);
  return;
 }
}

//我們進到這個方法中
@Nullable
 public static BeanDefinition registerAutoProxyCreatorIfNecessary(
   BeanDefinitionRegistry registry, @Nullable Object source) {
        //可以看到引入了InfrastructureAdvisorAutoProxyCreator這個類,那么這個類又是什么呢
  return registerOrEscalateApcAsRequired(InfrastructureAdvisorAutoProxyCreator.class, registry, source);
 }

//先看一下
public class InfrastructureAdvisorAutoProxyCreator extends AbstractAdvisorAutoProxyCreator {

 @Nullable
 private ConfigurableListableBeanFactory beanFactory;

 @Override
 protected void initBeanFactory(ConfigurableListableBeanFactory beanFactory) {
  super.initBeanFactory(beanFactory);
  this.beanFactory = beanFactory;
 }

 @Override
 protected boolean isEligibleAdvisorBean(String beanName) {
  return (this.beanFactory != null && this.beanFactory.containsBeanDefinition(beanName) &&
    this.beanFactory.getBeanDefinition(beanName).getRole() == BeanDefinition.ROLE_INFRASTRUCTURE);
 }
}

看一下繼承結(jié)構(gòu)圖

可以看到這個方法間接繼承于SmartInstantiationAwareBeanPostProcessor,最終繼承于BeanPostProcessor,這說明InfrastructureAdvisorAutoProxyCreator類是一個后置處理器,并且跟 spring AOP 開啟@EnableAspectJAutoProxy 時注冊的AnnotationAwareAspectJProxyCreator實現(xiàn)的是同⼀個接口,這也對應(yīng)了我之前所說聲明式事務(wù)是springAOP思想的一種應(yīng)用。

然后我們回過頭來再看ProxyTransactionManagementConfiguration這個類,我們看到其中有一個事務(wù)增強器,一個屬性解析器和是一個事務(wù)攔截器

    @Bean(name = TransactionManagementConfigUtils.TRANSACTION_ADVISOR_BEAN_NAME)
 @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
 public BeanFactoryTransactionAttributeSourceAdvisor transactionAdvisor(
   TransactionAttributeSource transactionAttributeSource, TransactionInterceptor transactionInterceptor) {

  // 事務(wù)增強器
  BeanFactoryTransactionAttributeSourceAdvisor advisor = new BeanFactoryTransactionAttributeSourceAdvisor();
  // 注入屬性解析器
  advisor.setTransactionAttributeSource(transactionAttributeSource);
  // 注入事務(wù)攔截器
  advisor.setAdvice(transactionInterceptor);
  if (this.enableTx != null) {
   advisor.setOrder(this.enableTx.<Integer>getNumber("order"));
  }
  return advisor;
 }

    @Bean
 @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
 // 屬性解析器
 public TransactionAttributeSource transactionAttributeSource() {
  return new AnnotationTransactionAttributeSource();
 }

 @Bean
 @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
 // 事務(wù)攔截器
 public TransactionInterceptor transactionInterceptor(TransactionAttributeSource transactionAttributeSource) {
  TransactionInterceptor interceptor = new TransactionInterceptor();
  interceptor.setTransactionAttributeSource(transactionAttributeSource);
  if (this.txManager != null) {
   interceptor.setTransactionManager(this.txManager);
  }
  return interceptor;
 }

先看看屬性解析器

//注解解析器集合
 private final Set<TransactionAnnotationParser> annotationParsers;

這是一個注解解析器的集合,可以添加多種注解解析器,在這里我們主要關(guān)注的是spring事務(wù)注解解析器SpringTransactionParser,看一下相關(guān)代碼

protected TransactionAttribute parseTransactionAnnotation(AnnotationAttributes attributes) {
  RuleBasedTransactionAttribute rbta = new RuleBasedTransactionAttribute();

  // 對應(yīng)Transaction注解的相關(guān)屬性
  Propagation propagation = attributes.getEnum("propagation");
  rbta.setPropagationBehavior(propagation.value());
  Isolation isolation = attributes.getEnum("isolation");
  rbta.setIsolationLevel(isolation.value());
  rbta.setTimeout(attributes.getNumber("timeout").intValue());
  rbta.setReadOnly(attributes.getBoolean("readOnly"));
  rbta.setQualifier(attributes.getString("value"));

  List<RollbackRuleAttribute> rollbackRules = new ArrayList<>();
  for (Class<?> rbRule : attributes.getClassArray("rollbackFor")) {
   rollbackRules.add(new RollbackRuleAttribute(rbRule));
  }
  for (String rbRule : attributes.getStringArray("rollbackForClassName")) {
   rollbackRules.add(new RollbackRuleAttribute(rbRule));
  }
  for (Class<?> rbRule : attributes.getClassArray("noRollbackFor")) {
   rollbackRules.add(new NoRollbackRuleAttribute(rbRule));
  }
  for (String rbRule : attributes.getStringArray("noRollbackForClassName")) {
   rollbackRules.add(new NoRollbackRuleAttribute(rbRule));
  }
  rbta.setRollbackRules(rollbackRules);

  return rbta;
 }

可以看到這段代碼中的Enum和ClassArray其實正是@Transaction注解中的相關(guān)屬性,這個屬性解析器的作用之一就是用來解析@Transaction注解中的屬性
看完了屬性解析器,我們接下來看事務(wù)攔截器TransactionInterceptor,其中重要的是這段代碼

    @Override
 @Nullable
 public Object invoke(MethodInvocation invocation) throws Throwable {
  // Work out the target class: may be {@code null}.
  // The TransactionAttributeSource should be passed the target class
  // as well as the method, which may be from an interface.
  Class<?> targetClass = (invocation.getThis() != null ? AopUtils.getTargetClass(invocation.getThis()) : null);

  // Adapt to TransactionAspectSupport's invokeWithinTransaction...
  // 增加事務(wù)支持
  return invokeWithinTransaction(invocation.getMethod(), targetClass, invocation::proceed);
 }

然后我們進到這個方法里面

@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.
  // 獲取屬性解析器,在配置類ProxyTransactionManagementConfiguration配置時加入
  TransactionAttributeSource tas = getTransactionAttributeSource();
  final TransactionAttribute txAttr = (tas != null ? tas.getTransactionAttribute(method, targetClass) : null);
  final TransactionManager tm = determineTransactionManager(txAttr);

  if (this.reactiveAdapterRegistry != null && tm instanceof ReactiveTransactionManager) {
   ReactiveTransactionSupport txSupport = this.transactionSupportCache.computeIfAbsent(method, key -> {
    if (KotlinDetector.isKotlinType(method.getDeclaringClass()) && KotlinDelegate.isSuspend(method)) {
     throw new TransactionUsageException(
       "Unsupported annotated transaction on suspending function detected: " + method +
       ". Use TransactionalOperator.transactional extensions instead.");
    }
    ReactiveAdapter adapter = this.reactiveAdapterRegistry.getAdapter(method.getReturnType());
    if (adapter == null) {
     throw new IllegalStateException("Cannot apply reactive transaction to non-reactive return type: " +
       method.getReturnType());
    }
    return new ReactiveTransactionSupport(adapter);
   });
   return txSupport.invokeWithinTransaction(
     method, targetClass, invocation, txAttr, (ReactiveTransactionManager) tm);
  }
  // 獲取事務(wù)管理器
  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 {
    // This is an around advice: Invoke the next interceptor in the chain.
    // This will normally result in a target object being invoked.
    retVal = invocation.proceedWithInvocation();
   }
   catch (Throwable ex) {
    // target invocation exception
    // 目標方法拋異常,會執(zhí)行回滾的操作
    completeTransactionAfterThrowing(txInfo, ex);
    throw ex;
   }
   finally {
    cleanupTransactionInfo(txInfo);
   }

   if (vavrPresent && VavrDelegate.isVavrTry(retVal)) {
    // Set rollback-only in case of Vavr failure matching our rollback rules...
    TransactionStatus status = txInfo.getTransactionStatus();
    if (status != null && txAttr != null) {
     retVal = VavrDelegate.evaluateTryFailure(retVal, txAttr, status);
    }
   }
   // 目標方法正常運行,會執(zhí)行commitTransactionAfterReturning,執(zhí)行事務(wù)提交操作
   commitTransactionAfterReturning(txInfo);
   return retVal;
  }

  else {
   final ThrowableHolder throwableHolder = new ThrowableHolder();

   // It's a CallbackPreferringPlatformTransactionManager: pass a TransactionCallback in.
   try {
    Object result = ((CallbackPreferringPlatformTransactionManager) ptm).execute(txAttr, status -> {
     TransactionInfo txInfo = prepareTransactionInfo(ptm, txAttr, joinpointIdentification, status);
     try {
      Object retVal = invocation.proceedWithInvocation();
      if (vavrPresent && VavrDelegate.isVavrTry(retVal)) {
       // Set rollback-only in case of Vavr failure matching our rollback rules...
       retVal = VavrDelegate.evaluateTryFailure(retVal, txAttr, status);
      }
      return retVal;
     }
     catch (Throwable ex) {
      if (txAttr.rollbackOn(ex)) {
       // A RuntimeException: will lead to a rollback.
       if (ex instanceof RuntimeException) {
        throw (RuntimeException) ex;
       }
       else {
        throw new ThrowableHolderException(ex);
       }
      }
      else {
       // A normal return value: will lead to a commit.
       throwableHolder.throwable = ex;
       return null;
      }
     }
     finally {
      cleanupTransactionInfo(txInfo);
     }
    });

    // Check result state: It might indicate a Throwable to rethrow.
    if (throwableHolder.throwable != null) {
     throw throwableHolder.throwable;
    }
    return result;
   }
   catch (ThrowableHolderException ex) {
    throw ex.getCause();
   }
   catch (TransactionSystemException ex2) {
    if (throwableHolder.throwable != null) {
     logger.error("Application exception overridden by commit exception", throwableHolder.throwable);
     ex2.initApplicationException(throwableHolder.throwable);
    }
    throw ex2;
   }
   catch (Throwable ex2) {
    if (throwableHolder.throwable != null) {
     logger.error("Application exception overridden by commit exception", throwableHolder.throwable);
    }
    throw ex2;
   }
  }
 }

總結(jié)

總體來說,spring實現(xiàn)聲明式事務(wù)的過程是這樣的

  • @EnableTransactionManagement 注解,通過@import引⼊了TransactionManagementConfigurationSelector類,它的selectImports⽅法導⼊了另外兩個類:AutoProxyRegistrar和ProxyTransactionManagementConfiguration
  • AutoProxyRegistrar類中方法registerBeanDefinitions中,通過 AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry)引⼊InfrastructureAdvisorAutoProxyCreator,是一個后置處理器類
  • ProxyTransactionManagementConfiguration 是⼀個添加了@Configuration注解的配置類,注冊了事務(wù)增強器(注⼊屬性解析器、事務(wù)攔截器)AnnotationTransactionAttributeSource和TransactionInterceptor,AnnotationTransactionAttributeSource內(nèi)部持有了⼀個解析器集合 Set annotationParsers,具體使⽤的是SpringTransactionAnnotationParser解析器,用來解析@Transactional的事務(wù)屬性,事務(wù)攔截器TransactionInterceptor實現(xiàn)了MethodInterceptor接⼝,該通用攔截會在產(chǎn)⽣代理對象之前和aop增強合并,最終⼀起影響到代理對象,TransactionInterceptor的invoke⽅法中invokeWithinTransaction會觸發(fā)原有業(yè)務(wù)邏輯調(diào)用(增強事務(wù))

到此這篇關(guān)于spring是如何實現(xiàn)聲明式事務(wù)的的文章就介紹到這了,更多相關(guān)spring 聲明式事務(wù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java前后端時間格式的轉(zhuǎn)化方式

    Java前后端時間格式的轉(zhuǎn)化方式

    這篇文章主要介紹了Java前后端時間格式的轉(zhuǎn)化方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • 淺談Apache Maven ToolChains的使用

    淺談Apache Maven ToolChains的使用

    Maven是java中非常有用和常用的構(gòu)建工具,基本上現(xiàn)在大型的java項目都是Maven和gradle的天下了。本文將介紹Apache Maven ToolChains的使用。
    2021-06-06
  • java中的類型自動轉(zhuǎn)換機制解析

    java中的類型自動轉(zhuǎn)換機制解析

    這篇文章主要介紹了java中的類型自動轉(zhuǎn)換機制,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • java實現(xiàn)聯(lián)機五子棋

    java實現(xiàn)聯(lián)機五子棋

    這篇文章主要為大家詳細介紹了java實現(xiàn)聯(lián)機五子棋,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-05-05
  • Java實現(xiàn)的簡單音樂播放器功能示例

    Java實現(xiàn)的簡單音樂播放器功能示例

    這篇文章主要介紹了Java實現(xiàn)的簡單音樂播放器功能,涉及java針對多媒體文件相關(guān)載入、播放相關(guān)操作技巧,需要的朋友可以參考下
    2019-02-02
  • JDK8中新增的Optional工具類基本使用

    JDK8中新增的Optional工具類基本使用

    Optional不是對null關(guān)鍵字的一種替代,而是對于null判定提供了一種更加優(yōu)雅的實現(xiàn),接下來通過本文給大家分享JDK8中新增的Optional工具類基本使用,感興趣的朋友跟隨小編一起看看吧
    2021-06-06
  • MyBatis傳入集合 list 數(shù)組 map參數(shù)的寫法

    MyBatis傳入集合 list 數(shù)組 map參數(shù)的寫法

    這篇文章主要介紹了MyBatis傳入集合 list 數(shù)組 map參數(shù)的寫法的相關(guān)資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-06-06
  • Java多線程處理List問題

    Java多線程處理List問題

    這篇文章主要介紹了Java多線程處理List問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • Java如何獲取對象屬性及對應(yīng)值

    Java如何獲取對象屬性及對應(yīng)值

    這篇文章主要介紹了Java如何獲取對象屬性及對應(yīng)值,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-11-11
  • Java中的Cookie和Session詳細解析

    Java中的Cookie和Session詳細解析

    這篇文章主要介紹了Java中的Cookie和Session詳細解析,客戶端會話技術(shù),服務(wù)端給客戶端的數(shù)據(jù),存儲于客戶端(瀏覽器),由于是保存在客戶端上的,所以存在安全問題,需要的朋友可以參考下
    2024-01-01

最新評論