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

被折叠的 条评论
为什么被折叠?



