探索Java中private方法添加@Transactional事务未生效原因

网友投稿 360 2022-09-13


探索Java中private方法添加@Transactional事务未生效原因

现在产品期望用户创建和保存逻辑分离:把User实例的创建和保存逻辑拆到两个方法分别进行。然后,把事务的注解 @Transactional 加在保存数据库的方法上。

@Service

public class StudentService {

@Autowired

private StudentMapper studentMapper;

@Autowired

private StudentService studentService;

public void saveStudent(String realname) throws Exception {

Student student = new Student();

student.setRealname(realname);

studentService.doSaveStudent(student);

}

@Transactional

private void doSaveStudent(Student student) throws Exception {

studentMapper.saveStudent(student);

if (student.getRealname().equals("小明")) {

throw new RuntimeException("该用户已存在");

}

}

}

执行程序,异常正常抛出

事务未回滚

源码解析

debug:

前一段是 Spring 创建 Bean 的过程。当 Bean 初始化之后,开始尝试代理操作,这是从如下方法开始处理的:

AbstractAutoProxyCreator#postProcessAfterInitialization

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;

}

继续 debug,直到

AopUtils#canApply

针对切面定义里的条件,确定这个方法是否可被应用创建成代理。有段 methodMatcher.matches(method, targetClass) 判断这个方法是否符合这样的条件:

public static boolean canApply(Pointcut pc, Class> targetClass, boolean hasIntroductionhttp://s) {

// ...

for (Class> clazz : classes) {

Method[] methods = ReflectionUtils.getAllDeclaredMethods(clazz);

for (Method method : methods) {

if (introductionAwareMethodMatcher != null ?

introductionAwareMethodMatcher.matches(method, targetClass, hasIntroductions) :

methodMatcherhttp://.matches(method, targetClass)) {

return true;

}

}

}

return false;

}

从 matches() 调用到

AbstractFallbackTransactionAttributeSource#getTransactionAttribute

获取注解中的事务属性,根据属性确定事务的策略。

接着调用到

computeTransactionAttribute

根据方法和类的类型确定是否返回事务属性:

当上图中条件判断结果为 true,则返回 null,表明该方法不会被代理,从而导致事务注解不会生效。

那到底是不是 true 呢?

条件1:allowPublicMethodsOnly()

AnnotationTransactionAttributeSource#publicMethodsOnly属性值

publicMethodsOnly 是通过 AnnotationTransactionAttributeSource 的构造方法初始化http://的,默认为 true。

条件2:Modifier.isPublic()

根据传入的 method.getModifiers() 获取方法的修饰符,该修饰符是 java.lang.reflect.Modifier 的静态属性,对应的几类修饰符分别是:

PUBLIC: 1

PRIVATE: 2

PROTECTED: 4

这里做了一个位运算,只有当传入的方法修饰符是 public 类型的时候,才返回 true

综上两个条件,只有当注解为事务方法为 public 才会被 Spring 处理。

修正

只需将修饰符从 private 改成 public,其实该问题 IDEA 也会告警,一般都会避免。

调用这个加了事务注解的方法,必须是调用被 Spring AOP 代理过的方法:不能通过类的内部调用或通过 this 调用。所以我们的案例的StudentService,它Autowired了自身(StudentService)的一个实例来完成代理方法的调用。


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:IS-IS 中间系统到中间系统
下一篇:10个提升工作效率的Secure CRT小窍门 - 你玩转了几个?(工作中最重要的提高效率)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~