Spring事務(wù)的開啟原理詳解
代碼如下:
@Configuration@EnableTransactionManagementpublic class RootConfig{ @Bean public DataSource dataSource(){ DruidDataSource dataSource = new DruidDataSource(); dataSource.setXXX(); ... return dataSource; } @Bean public PlatfromTransactionManager txManager(){ return new DataSourceTransactionManager(dataSource()); }}
@Servicepublic class UserService{ @Autowired private UserRepository userRepository; @Transactional public void addUser(User user){ userRepository.save(user); }}@EnableTransactionManagement開啟事務(wù)原理解析
@EnableTransactionManagement源碼如下:
@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Import(TransactionManagementConfigurationSelector.class)public @interface EnableTransactionManagement { boolean proxyTargetClass() default false; AdviceMode mode() default AdviceMode.PROXY; int order() default Ordered.LOWEST_PRECEDENCE;}
可以看到,@EnableTransactionManagement接口類主要Import了TransactionManagementConfigurationSelector來實(shí)現(xiàn)其注入,而TransactionManagementConfigurationSelector又主要使用selectImport方法來實(shí)現(xiàn)其注入,代碼如下:
@Overridepublic final String[] selectImports(AnnotationMetadata importingClassMetadata) { Class<?> annoType = GenericTypeResolver.resolveTypeArgument(getClass(), AdviceModeImportSelector.class); AnnotationAttributes attributes = AnnotationConfigUtils.attributesFor(importingClassMetadata, annoType); if (attributes == null) { throw new IllegalArgumentException(String.format( '@%s is not present on importing class ’%s’ as expected', annoType.getSimpleName(), importingClassMetadata.getClassName())); } AdviceMode adviceMode = attributes.getEnum(this.getAdviceModeAttributeName()); //根據(jù)AdviceMode返回不同的類型,默認(rèn)是AdviceMode.PROXY。 String[] imports = selectImports(adviceMode); if (imports == null) { throw new IllegalArgumentException(String.format('Unknown AdviceMode: ’%s’', adviceMode)); } return imports;}@Overrideprotected 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; }}
其中主要功能點(diǎn)為根據(jù)AdviceMode選擇創(chuàng)建不同的bean,AdviceMode的默認(rèn)代理方式是PROXY,jdk代理。所以返回的是AutoProxyRegistrar和ProxyTransactionManagementConfiguration。
我們先分析AutoProxyRegistrar,AutoProxyRegistrar實(shí)現(xiàn)了ImportBeanDefinitionRegistrar,那在創(chuàng)建bean的時(shí)候會(huì)調(diào)用registerBeanDefinitions方法。registerBeanDefinitions方法的實(shí)現(xiàn):
@Overridepublic void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) { boolean candidateFound = false; Set<String> annoTypes = importingClassMetadata.getAnnotationTypes(); for (String annoType : annoTypes) { AnnotationAttributes candidate = AnnotationConfigUtils.attributesFor(importingClassMetadata, annoType); if (candidate == null) { continue; } Object mode = candidate.get('mode'); Object proxyTargetClass = candidate.get('proxyTargetClass'); if (mode != null && proxyTargetClass != null && AdviceMode.class == mode.getClass() && Boolean.class == proxyTargetClass.getClass()) { candidateFound = true; //只有@EnableTransactionManagement注解才會(huì)走到這里 if (mode == AdviceMode.PROXY) { AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry); if ((Boolean) proxyTargetClass) { AopConfigUtils.forceAutoProxyCreatorToUseClassProxying(registry); return; } } } } //...}public static BeanDefinition registerAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry, Object source) { return registerOrEscalateApcAsRequired(InfrastructureAdvisorAutoProxyCreator.class, registry, source);}
可以看到,它通過注冊(cè)InfrastructureAdvisorAutoProxyCreator來啟動(dòng)Spring Aop。
接下來再看ProxyTransactionManagementConfiguration的作用,代碼如下:
@Configurationpublic class ProxyTransactionManagementConfiguration extends AbstractTransactionManagementConfiguration { @Bean(name = TransactionManagementConfigUtils.TRANSACTION_ADVISOR_BEAN_NAME) @Role(BeanDefinition.ROLE_INFRASTRUCTURE) public BeanFactoryTransactionAttributeSourceAdvisor transactionAdvisor() { BeanFactoryTransactionAttributeSourceAdvisor advisor = new BeanFactoryTransactionAttributeSourceAdvisor(); advisor.setTransactionAttributeSource(transactionAttributeSource()); advisor.setAdvice(transactionInterceptor()); 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) public TransactionInterceptor transactionInterceptor() { TransactionInterceptor interceptor = new TransactionInterceptor(); interceptor.setTransactionAttributeSource(transactionAttributeSource()); if (this.txManager != null) { interceptor.setTransactionManager(this.txManager); } return interceptor; }}
ProxyTransactionManagementConfiguration是一個(gè)配置文件,注冊(cè)了三個(gè)bean,BeanFactoryTransactionAttributeSourceAdvisor、AnnotationTransactionAttributeSource、TransactionInterceptor,而這三個(gè)類分別繼承Advisor、Advice和Pointcut。即切面所需組件。
總結(jié)@EnableTransactionManagement利用AutoProxyRegistrar啟動(dòng)Spring Aop,使用ProxyTransactionManagementConfiguration配置對(duì)應(yīng)切面部件。
以上就是Spring事務(wù)的簡(jiǎn)單實(shí)現(xiàn)步驟的詳細(xì)內(nèi)容,更多關(guān)于Spring事務(wù)實(shí)現(xiàn)步驟的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. 解決Python 進(jìn)程池Pool中一些坑2. 三個(gè)不常見的 HTML5 實(shí)用新特性簡(jiǎn)介3. Python獲取抖音關(guān)注列表封號(hào)賬號(hào)的實(shí)現(xiàn)代碼4. Python使用jupyter notebook查看ipynb文件過程解析5. ajax請(qǐng)求添加自定義header參數(shù)代碼6. python利用os模塊編寫文件復(fù)制功能——copy()函數(shù)用法7. Python如何讀寫CSV文件8. php網(wǎng)絡(luò)安全中命令執(zhí)行漏洞的產(chǎn)生及本質(zhì)探究9. php測(cè)試程序運(yùn)行速度和頁(yè)面執(zhí)行速度的代碼10. 無線標(biāo)記語(yǔ)言(WML)基礎(chǔ)之WMLScript 基礎(chǔ)第1/2頁(yè)
