声明式事务(Transactional)的工作原理

Spring Boot通过起步依赖和自动配置简化了开发环境的配置,节约了开发成本。但是作为一名技术人员,如果不了解一下底层的实现原理,未免太过肤浅。

声明式事物我们一直在用,但是除了@Transactional注解的使用方式,具体底层是如何支撑的,实现原理是什么你真的知道吗?

1. 前置条件(pre-condition)

要实现Transaction的自动配置,需要执行以下自动配置路径
DataSourceAutoConfiguration -> DataSourceTransactionManagerAutoConfiguration -> TransactionAutoConfiguration

DataSourceAutoConfiguration可以配置数据源、数据库连接池等信息。

DataSourceTransactionManagerAutoConfiguration配置了数据源事物管理器(DataSourceTransactionManager),DataSourceTransactionManager的类图如下,它实现了PlatformTransactionManager接口。
在这里插入图片描述

2. Transaction自动配置(TransactionAutoConfiguration)

Transaction自动配置实现了一下功能:

  1. 个性化PlatformTransactionManager,如上述的DataSourceTransactionManager
  2. 基于PlatformTransactionManager创建TransactionTemplate
  3. 通过AOP代理实现事务管理,默认配置下,SpringBoot默认启用的是Cglib事务代理。

org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration.EnableTransactionManagementConfiguration

@Configuration
@ConditionalOnClass(PlatformTransactionManager.class)
@AutoConfigureAfter({ JtaAutoConfiguration.class, HibernateJpaAutoConfiguration.class,
		DataSourceTransactionManagerAutoConfiguration.class,
		Neo4jDataAutoConfiguration.class })
@EnableConfigurationProperties(TransactionProperties.class)
public class TransactionAutoConfiguration {

	@Bean
	@ConditionalOnMissingBean
	public TransactionManagerCustomizers platformTransactionManagerCustomizers(
			ObjectProvider<PlatformTransactionManagerCustomizer<?>> customizers) {
		return new TransactionManagerCustomizers(
				customizers.orderedStream().collect(Collectors.toList()));
	}

	@Configuration
	@ConditionalOnSingleCandidate(PlatformTransactionManager.class)
	public static class TransactionTemplateConfiguration {

		private final PlatformTransactionManager transactionManager;

		public TransactionTemplateConfiguration(
				PlatformTransactionManager transactionManager) {
			this.transactionManager = transactionManager;
		}

		@Bean
		@ConditionalOnMissingBean
		public TransactionTemplate transactionTemplate() {
			return new TransactionTemplate(this.transactionManager);
		}

	}

	@Configuration
	@ConditionalOnBean(PlatformTransactionManager.class)
	@ConditionalOnMissingBean(AbstractTransactionManagementConfiguration.class)
	public static class EnableTransactionManagementConfiguration {

		@Configuration
		@EnableTransactionManagement(proxyTargetClass = false)
		@ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class",
				havingValue = "false", matchIfMissing = false)
		public static class JdkDynamicAutoProxyConfiguration {

		}

		@Configuration
		@EnableTransactionManagement(proxyTargetClass = true)
		@ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class",
				havingValue = "true", matchIfMissing = true)
		public static class CglibAutoProxyConfiguration {

		}
	}
}

通过AOP代理实现事务管理通过注解@EnableTransactionManagement实现。
首先,它通过Import注解引入了TransactionManagementConfigurationSelector,它是一个ImportSelector实现。
在代理模式下他会返回 AutoProxyRegistrar 和 ProxyTransactionManagementConfiguration。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import(TransactionManagementConfigurationSelector.class)
public @interface EnableTransactionManagement {
public class TransactionManagementConfigurationSelector extends AdviceModeImportSelector<EnableTransactionManagement> {
	/**
	 * Returns {@link ProxyTransactionManagementConfiguration} or
	 * {@code AspectJ(Jta)TransactionManagementConfiguration} for {@code PROXY}
	 * and {@code ASPECTJ} values of {@link EnableTransactionManagement#mode()},
	 * respectively.
	 */
	@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;
		}
	}

	private String determineTransactionAspectClass() {
		return (ClassUtils.isPresent("javax.transaction.Transactional", getClass().getClassLoader()) ?
				TransactionManagementConfigUtils.JTA_TRANSACTION_ASPECT_CONFIGURATION_CLASS_NAME :
				TransactionManagementConfigUtils.TRANSACTION_ASPECT_CONFIGURATION_CLASS_NAME);
	}
}

2.1 AutoProxyRegistrar -> InfrastructureAdvisorAutoProxyCreator

这是一个ImportBeanDefinitionRegistrar类,它引入了一个类型为InfrastructureAdvisorAutoProxyCreator的BeanDefinition,InfrastructureAdvisorAutoProxyCreator是BeanPostProcessor 的实现类,通过它可以在Bean实例化前后创建AOP代理类。

2.1.1 postProcessBeforeInstantiation

如果指定了个性化TargetSource,会在实例化之前创建代理类。
org.springframework.aop.framework.autoproxy.AbstractAutoProxyCreator#postProcessBeforeInstantiation

@Override
public Object postProcessBeforeInstantiation(Class<?> beanClass, String beanName) {
	Object cacheKey = getCacheKey(beanClass, beanName);

	if (!StringUtils.hasLength(beanName) || !this.targetSourcedBeans.contains(beanName)) {
		if (this.advisedBeans.containsKey(cacheKey)) {
			return null;
		}
		if (isInfrastructureClass(beanClass) || shouldSkip(beanClass, beanName)) {
			this.advisedBeans.put(cacheKey, Boolean.FALSE);
			return null;
		}
	}

	// Create proxy here if we have a custom TargetSource.
	// Suppresses unnecessary default instantiation of the target bean:
	// The TargetSource will handle target instances in a custom fashion.
	TargetSource targetSource = getCustomTargetSource(beanClass, beanName);
	if (targetSource != null) {
		if (StringUtils.hasLength(beanName)) {
			this.targetSourcedBeans.add(beanName);
		}
		Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(beanClass, beanName, targetSource);
		Object proxy = createProxy(beanClass, beanName, specificInterceptors, targetSource);
		this.proxyTypes.put(cacheKey, proxy.getClass());
		return proxy;
	}

	return null;
}

2.1.2 postProcessAfterInitialization

所有的Bean实例化后都会执行postProcessAfterInitialization方法;如果需要则创建代理类。

/**
 * Create a proxy with the configured interceptors if the bean is
 * identified as one to proxy by the subclass.
 * @see #getAdvicesAndAdvisorsForBean
 */
@Override
public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) {
	if (bean != null) {
		Object cacheKey = getCacheKey(bean.getClass(), beanName);
		if (this.earlyProxyReferences.remove(cacheKey) != bean) {
			return wrapIfNecessary(bean, beanName, cacheKey);
		}
	}
	return bean;
}

判断是否需要创建代理类的逻辑:
具体代码可参考org.springframework.aop.support.AopUtils#canApply(org.springframework.aop.Pointcut, java.lang.Class<?>, boolean)

  1. 获取所有的类型为 Advisor 的Bean Definition 列表,并获取 Bean 实例;
  2. 如果 Advisor 是 IntroductionAdvisor 的子类,通过 ClassFilter 匹配是否需要代理类,如果匹配返回 TRUE;
  3. 如果 Advisor 是 PointcutAdvisor 的子类,通过 ClassFilter 进行匹配,如果不匹配直接返回 FALSE;
  4. 获取目标类以及其所有接口类,校验所有的方法是否满足 MethodMatcher 匹配,只要有满足匹配的方法返回 TRUE;

支持事务的Advisor类为:BeanFactoryTransactionAttributeSourceAdvisor,它是PointcutAdvisor的子类,其注入的切点类为
org.springframework.transaction.interceptor.TransactionAttributeSourcePointcut。

创建代理类的逻辑:新建ProxyFactory对象,设置属性,通过getProxy方法获取代理类。
org.springframework.aop.framework.autoproxy.AbstractAutoProxyCreator#createProxy

protected Object createProxy(Class<?> beanClass, @Nullable String beanName,
	@Nullable Object[] specificInterceptors, TargetSource targetSource) {

	if (this.beanFactory instanceof ConfigurableListableBeanFactory) {
		AutoProxyUtils.exposeTargetClass((ConfigurableListableBeanFactory) this.beanFactory, beanName, beanClass);
	}

	ProxyFactory proxyFactory = new ProxyFactory();
	proxyFactory.copyFrom(this);

	if (!proxyFactory.isProxyTargetClass()) {
		if (shouldProxyTargetClass(beanClass, beanName)) {
			proxyFactory.setProxyTargetClass(true);
		}
		else {
			evaluateProxyInterfaces(beanClass, proxyFactory);
		}
	}

	Advisor[] advisors = buildAdvisors(beanName, specificInterceptors);
	proxyFactory.addAdvisors(advisors);
	proxyFactory.setTargetSource(targetSource);
	customizeProxyFactory(proxyFactory);

	proxyFactory.setFrozen(this.freezeProxy);
	if (advisorsPreFiltered()) {
		proxyFactory.setPreFiltered(true);
	}

	return proxyFactory.getProxy(getProxyClassLoader());
}

创建AopProxy,这里最终创建的是ObjenesisCglibAopProxy,然后调用AopProxy的getProxy方法获取代理类。

@Override
public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
	if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
		Class<?> targetClass = config.getTargetClass();
		if (targetClass == null) {
			throw new AopConfigException("TargetSource cannot determine target class: " +
					"Either an interface or a target is required for proxy creation.");
		}
		if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) {
			return new JdkDynamicAopProxy(config);
		}
		return new ObjenesisCglibAopProxy(config);
	}
	else {
		return new JdkDynamicAopProxy(config);
	}
}

2.2 注册Advisor -> ProxyTransactionManagementConfiguration

该Configuration类注册了一个类型为BeanFactoryTransactionAttributeSourceAdvisor的BeanDefinition。

该Advisor的主要作用是为所有基于@Transactional注解的方法创建AOP代理,以便增加事务控制,事务控制通过TransactionInterceptor来实现。

@Configuration
public 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());
		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)
	public TransactionInterceptor transactionInterceptor() {
		TransactionInterceptor interceptor = new TransactionInterceptor();
		interceptor.setTransactionAttributeSource(transactionAttributeSource());
		if (this.txManager != null) {
			interceptor.setTransactionManager(this.txManager);
		}
		return interceptor;
	}
}

3. 代理类执行事务的逻辑 - TransactionInterceptor

org.springframework.transaction.interceptor.TransactionAspectSupport#invokeWithinTransaction

@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 PlatformTransactionManager tm = determineTransactionManager(txAttr);
	final String joinpointIdentification = methodIdentification(method, targetClass, txAttr);

	if (txAttr == null || !(tm instanceof CallbackPreferringPlatformTransactionManager)) {
		// Standard transaction demarcation with getTransaction and commit/rollback calls.
		TransactionInfo txInfo = createTransactionIfNecessary(tm, 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
			completeTransactionAfterThrowing(txInfo, ex);
			throw ex;
		}
		finally {
			cleanupTransactionInfo(txInfo);
		}
		commitTransactionAfterReturning(txInfo);
		return retVal;
	}

	else {
		final ThrowableHolder throwableHolder = new ThrowableHolder();

		// It's a CallbackPreferringPlatformTransactionManager: pass a TransactionCallback in.
		try {
			Object result = ((CallbackPreferringPlatformTransactionManager) tm).execute(txAttr, status -> {
				TransactionInfo txInfo = prepareTransactionInfo(tm, txAttr, joinpointIdentification, status);
				try {
					return invocation.proceedWithInvocation();
				}
				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;
		}
	}
}

4. 事务同步管理

Spring开始事务后会将连接信息保存到ThreadLocal内以备后续处理使用,以此来保证事务的一致性。整个过程通过类TransactionSynchronizationManager来实现。

org.springframework.jdbc.datasource.DataSourceTransactionManager#doBegin

@Override
protected void doBegin(Object transaction, TransactionDefinition definition) {
	DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;
	Connection con = null;

	try {
		if (!txObject.hasConnectionHolder() ||
				txObject.getConnectionHolder().isSynchronizedWithTransaction()) {
			Connection newCon = obtainDataSource().getConnection();
			if (logger.isDebugEnabled()) {
				logger.debug("Acquired Connection [" + newCon + "] for JDBC transaction");
			}
			txObject.setConnectionHolder(new ConnectionHolder(newCon), true);
		}

		txObject.getConnectionHolder().setSynchronizedWithTransaction(true);
		con = txObject.getConnectionHolder().getConnection();

		Integer previousIsolationLevel = DataSourceUtils.prepareConnectionForTransaction(con, definition);
		txObject.setPreviousIsolationLevel(previousIsolationLevel);

		// Switch to manual commit if necessary. This is very expensive in some JDBC drivers,
		// so we don't want to do it unnecessarily (for example if we've explicitly
		// configured the connection pool to set it already).
		if (con.getAutoCommit()) {
			txObject.setMustRestoreAutoCommit(true);
			if (logger.isDebugEnabled()) {
				logger.debug("Switching JDBC Connection [" + con + "] to manual commit");
			}
			con.setAutoCommit(false);
		}

		prepareTransactionalConnection(con, definition);
		txObject.getConnectionHolder().setTransactionActive(true);

		int timeout = determineTimeout(definition);
		if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
			txObject.getConnectionHolder().setTimeoutInSeconds(timeout);
		}

		// Bind the connection holder to the thread.
		if (txObject.isNewConnectionHolder()) {
			TransactionSynchronizationManager.bindResource(obtainDataSource(), txObject.getConnectionHolder());
		}
	}

	catch (Throwable ex) {
		if (txObject.isNewConnectionHolder()) {
			DataSourceUtils.releaseConnection(con, obtainDataSource());
			txObject.setConnectionHolder(null, false);
		}
		throw new CannotCreateTransactionException("Could not open JDBC Connection for transaction", ex);
	}
}

参考

spring源码阅读–@Transactional实现原理
深入理解Spring系列之十二:@Transactional是如何工作的

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: @Transactional 是 Spring 框架中用来声明事务的注解。它的原理是通过 AOP(面向切面编程)将需要进行事务管理的方法进行增强。当方法被调用时,Spring 会自动创建一个事务,并在方法执行前开启事务,方法执行后根据方法运行的结果决定是提交事务还是回滚事务。 当一个方法被声明为 @Transactional 时,Spring 会在运行时为该方法生成一个代理对象,该对象包含了事务管理的逻辑。当外部代码调用被 @Transactional 注解修饰的方法时,实际上是调用了代理对象的方法。 在代理对象的方法中,Spring 会先检查当前是否存在事务。如果存在事务,则直接使用现有的事务。如果不存在事务,则会创建一个新的事务。在方法执行前,Spring 会将事务的隔离级别设置到当前线程的 ThreadLocal 中,以便在方法执行过程中能够获取正确的隔离级别。 当方法执行完毕后,Spring 会检查方法的返回值。如果方法正常返回,则会提交事务,如果方法抛出异常,则会回滚事务。同时,Spring 也提供了通过 @Transactional(propagation = Propagation.REQUIRED) 设置事务的传播行为,以便支持在一个事务内调用另一个需要事务管理的方法,或者在没有事务的情况下执行。 总之,@Transactional原理是通过 AOP 技术实现的,可以在方法运行的过程中自动创建和管理事务,大大简化了事务管理的过程,并可以灵活地控制事务的隔离级别和传播行为。 ### 回答2: @transactional 是一个注解,用于在 Spring 框架中管理事务的声明式事务控制。它的原理主要涉及以下几个方面。 首先,@transactional 会被 Spring 容器解析并应用于被注解的方法或类。被注解的方法或类称为事务边界。一旦方法或类被标记为 @transactional,它们就具备了事务的特性。 其次,@transactional 使用了代理模式来实现事务的管理。Spring 使用 AOP(面向切面编程)的思想,将事务管理逻辑从业务逻辑中分离出来,通过代理对象包装被注解的方法或类,在方法调用之前或之后对事务逻辑进行处理。 接着,@transactional 的核心原理是使用了底层的事务管理器。Spring 为不同的持久层技术提供了不同的事务管理器实现,如 JDBC、Hibernate、JPA 等。这些事务管理器通过与数据库或其他持久层技术的交互来实现对事务的管理。 最后,@transactional 注解会根据方法的执行结果来决定是否将事务提交或回滚。如果方法执行成功,事务管理器会将事务提交到数据库,否则事务管理器会回滚事务,将数据库恢复到方法执行之前的状态。 总之,@transactional 注解的原理是通过代理模式和底层的事务管理器来实现对事务的管理,它能够保证被注解的方法或类具备事务的特性,并根据方法的执行结果进行事务的提交或回滚。这样可以简化事务管理的代码,提高系统的可靠性和性能。 ### 回答3: @Transactional 是Spring框架中的一个注解,用于管理数据库事务。其原理如下: 首先,@Transactional 注解可以在类级别或方法级别上使用。当应用程序调用被 @Transactional 注解标记的方法时,Spring会首先检查当前线程是否已经存在一个数据库事务。如果存在,则该事务将被使用,如果不存在,则将创建一个新的事务。 接着,Spring会将事务绑定到当前的数据库连接上,并在方法执行之前开始事务,并在方法执行完成后结束事务。如果方法执行期间出现异常,则事务将被回滚,否则事务将被提交。 在事务管理期间,Spring会使用AOP的代理机制对被 @Transactional 注解标记的方法进行拦截,在方法执行前、后和异常出现时应用相应的事务管理操作。 事务管理的核心是使用了数据库连接池,Spring会从连接池中获取连接,并在事务开始时将该连接绑定到当前线程上,当事务结束后,该连接将被释放回连接池。 除此之外,@Transactional 还支持多个事务传播行为,例如REQUIRED、SUPPORTS、REQUIRES_NEW等,可以根据业务需求进行配置。 总结来说,@Transactional原理是通过使用Spring AOP的拦截功能,在方法执行前、后和异常出现时对事务进行管理,并使用数据库连接池来获取和释放数据库连接,实现对数据库事务的管理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值