spring事务失效一:非public方法

本文探讨了Spring事务在非public方法上失效的原因。通过分析源码,揭示了Spring事务依赖动态代理,且仅对public方法生效。在初始化bean时,特定构造函数被调用,设置仅允许public方法生成代理对象,导致非public方法的事务失效。若修改源码,允许为非public方法生成代理,事务即可生效。

在使用spring事务的时候,如果使用不当,有可能会出现事务失效的问题,接下来,我基于自己的见解和学习,分享下我的看法

对于非public方法这种场景,我觉得这是最好理解的一种场景,@Transactional注解加在非public修饰的方法上,就不会生效,这个是没问题的,接下来,我分析下为什么非public方法会失效,其实在这篇博客中已经介绍了一部分:spring事务源码-代理对象生成过程解析

spring事务也是通过动态代理来实现的,spring事务也是借鉴了AOP的思想,在对一个bean进行初始化的过程中,在执行到第八个后置处理器方法,org.springframework.aop.framework.autoproxy.AbstractAutoProxyCreator#postProcessAfterInitialization
这里是怎么调用过去的,就不说了,在前面的笔记中有记录过,在这个方法中,会判断当前bean是否要生成代理对象,在判断的过程中,就和这里要说的非public判断有关系,在判断是否要生成代理对象的时候,会调用到这个方法org.springframework.transaction.interceptor.AbstractFallbackTransactionAttributeSource#computeTransactionAttribute

// Don't allow no-public methods as required.
if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) {
  return null;
}

可以看到,在方法刚进来的时候,就会有这么一个判断,第二个判断就简单了,就是判断当前method是否是public的,对于第一个判断

org.springframework.transaction.annotation.AnnotationTransactionAttributeSource#allowPublicMethodsOnly

/**
 * By default, only public methods can be made transactional.
 */
@Override
protected boolean allowPublicMethodsOnly() {
  return this.publicMethodsOnly;
}

其实就是返回了publicMethodsOnly这个变量,那既然是返回的这个变量,就看下在哪里对这个值进行了赋值,在AnnotationTransactionAttributeSource这个类中,进行全局搜索,会发现,这个变量只有在构造函数中赋值了,那就简单了,看下在哪里有调用构造方法就可以了
这个构造方法中,进行了赋值,其他的构造方法,默认publicMethodsOnly = true,所以我们要看下这个构造函数是在哪里被调用的

public AnnotationTransactionAttributeSource(boolean publicMethodsOnly) {
  this.publicMethodsOnly = publicMethodsOnly;
  if (jta12Present || ejb3Present) {
    this.annotationParsers = new LinkedHashSet<>(4);
    this.annotationParsers.add(new SpringTransactionAnnotationParser());
    if (jta12Present) {
      this.annotationParsers.add(new JtaTransactionAnnotationParser());
    }
    if (ejb3Present) {
      this.annotationParsers.add(new Ejb3TransactionAnnotationParser());
    }
  }
  else {
    this.annotationParsers = Collections.singleton(new SpringTransactionAnnotationParser());
  }
}

会发现,这个构造函数是在本类中的另外一个构造方法调用

public AnnotationTransactionAttributeSource() {
		this(true);
	}

那到这里,我们知道,spring的@EnableTransactionManager注解会注入三个bean,
在这里插入图片描述

看到这个截图,其实我觉得基本上就明了了
首先在注入这个bean的时候,调用的是无参构造函数,在无参构造函数中,指定了publicMethodsOnly为true,所以,如果我们不做任何改造的情况下,我们在使用事务的时候,是只允许public方法进行代理对象生成的
其实在没有看源码之前,我一直以为是在生成代理对象的时候,判断了是否是public方法,现在看来,其实不是的,jdk或者cglib都可以为非public方法生成代理对象,并不是我不给你生成,而是spring事务自己在调用动态代理之前,已经对非public方法过滤了,所以这是非public方法,事务不生效的原因
反而言之,如果我们修改了spring的源码,把默认的

public AnnotationTransactionAttributeSource() {
		this(true);
	}
修改为
public AnnotationTransactionAttributeSource() {
		this(false);
	}

那默认就可以为非public方法生成代理对象了

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值