Spring AOP无法拦截代理内部方法调用(同一个类中。一个没事务的方法,调用一个有事务方法,事务失效问题)

https://blog.csdn.net/sufu1065/article/details/122076645 事务失效的几种场景

https://blog.csdn.net/qq_24948625/article/details/124824624  失效原理

一下工具类解决 同一个类中。一个没事务的方法,调用一个有事务方法,事务失效问题

/**
 * 解决Spring AOP无法拦截代理内部方法调用
 */
@Component
public class SpringContextUtil implements ApplicationContextAware {

    private static ApplicationContext applicationContext;

    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        SpringContextUtil.applicationContext = applicationContext;
    }

    public static <T> T getBeanByName(String name) {
        return (T) applicationContext.getBean(name);
    }

    public static <T> T getBeanByClass(Class clazz) {
        return (T) applicationContext.getBean(clazz);
    }
}

  

 

 

原因:
声明式事务基于Spring AOP实现,将具体业务逻辑与事务处理解耦,在 Spring 的 AOP 代理下,只有目标方法由外部调用,目标方法才由 Spring 生成的代理对象来管理,这会造成自调用问题

spring的@Transactional事务生效的一个前提是进行方法调用前经过拦截器TransactionInterceptor,也就是说只有通过TransactionInterceptor拦截器的方法才会被加入到spring事务管理中

如果是在同一个类中的方法调用,则不会被方法拦截器拦截到,因此事务不会起作用,必须将方法放入另一个类,并且该类通过spring注入。

Transactional是Spring提供的事务管理注解。

重点在于,Spring采用动态代理(AOP)实现对bean的管理和切片,它为我们的每个class生成一个代理对象。只有在代理对象之间进行调用时,可以触发切面逻辑。

而在同一个class中,方法A调用方法B,调用的是原对象的方法,而不通过代理对象。所以Spring无法切到这次调用,也就无法通过注解保证事务性了。

也就是说,在同一个类中的方法调用,则不会被方法拦截器拦截到,因此事务不会起作用。

posted @ 2022-03-15 16:19  好记性不如烂笔头=>  阅读(736)  评论(0)    收藏  举报