Spring事务失效

6 篇文章 0 订阅

基础代码

事务配置类

@Configuration
@EnableTransactionManagement
@ComponentScan("life.cqq.transactional")
@EnableAspectJAutoProxy(proxyTargetClass = true, exposeProxy = true)
public class TxConfig {
    @Bean
    public DataSource dataSource() {
        DruidDataSource dataSource = new DruidDataSource();
        dataSource.setUsername("root");
        dataSource.setPassword("root");
        dataSource.setUrl("jdbc:mysql://localhost:3306/test?useSSL=false&serverTimezone=Asia/Shanghai");
        dataSource.setDriverClassName("com.mysql.jdbc.Driver");
        return dataSource;
    }

    @Bean
    public JdbcTemplate jdbcTemplate(DataSource dataSource) {
        return new JdbcTemplate(dataSource);
    }

    /**
     * 事务管理器
     */
    @Bean
    public PlatformTransactionManager platformTransactionManager() {
        return new DataSourceTransactionManager(dataSource());
    }
}

业务类

@Component
public class Service {

    @Autowired
    private JdbcTemplate jdbcTemplate;
    
    @Transactional(rollbackFor = Exception.class)
    public void insert() {
        jdbcTemplate.execute("INSERT INTO `test_table`(`username`, `password`) VALUES ('root1', 'root1')");
    }
}

测试类

public class TransactionalTest {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(TxConfig.class);
        Service service = context.getBean(Service.class);
        service.insert();
    }
}

前置问题

  1. 测试类中获取的service实例,是否有被代理?
  2. 当业务类中当insert方法的@Transactional注解删除时,测试类中获取的service实例,是否有被代理?

解:第一种情况在getBean时,获取的是代理对象。第二种情况会获取到原bean。

进而又引出了一个新问题,spring是如何根据方法的@Transactional注解生成的代理对象?

大胆推测,当bean在初始化后,会被一组BeanPostProcessor执行后置处理。那么,一定有一个针对于事务的Bean后置处理器,将bean转为代理后的bean。经过追踪源码,推测的结果是正确的。
简单来说:BeanPostProcessor: AnnotationAwareAspectJAutoProxyCreator 判断bean是否有能被computeTransactionAttribute方法计算出事务属性的方法,有则生成关于事务的代理类。

事务失效情景

一、无事务注解方法中调用有事务注解方法

@Component
public class Service {

    @Autowired
    private JdbcTemplate jdbcTemplate;
    
    public void insert() {
        jdbcTemplate.execute("INSERT INTO `test_table`(`username`, `password`) VALUES ('root1', 'root1')");
        insert2();
        throw new NullPointerException();
    }
    @Transactional(rollbackFor = Exception.class)
    public void insert2() {
        jdbcTemplate.execute("INSERT INTO `test_table`(`username`, `password`) VALUES ('root2', 'root2')");
    }
}

即使insert方法抛出了运行时异常,但两条数据都会被新增成功,不会被回滚。为什么?

首先,insert2添加了@Transactional注解,且无实现接口,那么getBean时获取到的是一个Cglib代理类,代理内容与事务相关。

private static class DynamicAdvisedInterceptor implements MethodInterceptor, Serializable {
    @Override
    @Nullable
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        // ....
        List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass);
        Object retVal;
        if (chain.isEmpty() && Modifier.isPublic(method.getModifiers())) {
            Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args);
            retVal = methodProxy.invoke(target, argsToUse);
        }
        else {
            // We need to create a method invocation...
            retVal = new CglibMethodInvocation(proxy, target, method, args, targetClass, chain, methodProxy).proceed();
        }
        // ....
    }
}

当通过代理类调用bean的方法时,会被拦截,判断方法前后是否需要执行事务代码,事务代码实际被封装在chain中的TransactionInterceptor中。如果判断方法不需要,chain的拦截器列表为空,则直接通过代理对象中的原Bean(proxy参数)调用bean方法。要理清一点:即使方法前后需要执行事务代码,但调用bean方法时,仍需要通过原Bean(proxy参数)进行调用。

那么由于我们调用的是没有事务注解的insert,getInterceptorsAndDynamicInterceptionAdvice时获取的chain为空,所以,当出现异常时,也没有进行事务回滚。

二、不同事务传播机制的方法间调用

@Component
public class Service {

    @Autowired
    private JdbcTemplate jdbcTemplate;

    @Transactional(rollbackFor = Exception.class)
    public void insert() {
        jdbcTemplate.execute("INSERT INTO `test_table`(`username`, `password`) VALUES ('root1', 'root1')");
        insert2();
    }
    @Transactional(rollbackFor = Exception.class, propagation = Propagation.NEVER)
    public void insert2() {
        jdbcTemplate.execute("INSERT INTO `test_table`(`username`, `password`) VALUES ('root2', 'root2')");
    }
}

以上代码,是可以正常结束的。并没有因为insert2的propagation配置的NEVER而抛出异常(NEVER简单来说为不在事务下执行,如果执行时已存在事务则抛出异常)为什么没有生效呢?

代理对象执行insert方法前执行intercept方法虽然获取到了事务拦截器,但是调用insert2方法时,并没有走intercept,而是通过内部的原Bean对象调用的。

如何解决呢,可以通过显示调用代理对象的方式调用insert2

@Transactional(rollbackFor = Exception.class)
public void insert() {
    jdbcTemplate.execute("INSERT INTO `test_table`(`username`, `password`) VALUES ('root1', 'root1')");
    // insert2();
    ((Service) AopContext.currentProxy()).insert2();
}

这样,执行insert2方法时,通过代理对象进行调用,会执行intercept方法判断并获取执行insert2方法时的事务拦截器。

又产生一个疑问,AopContext.currentProxy()是如何获取到的代理对象?

private static class DynamicAdvisedInterceptor implements MethodInterceptor, Serializable {
    Object oldProxy = null;
    boolean setProxyContext = false;
    Object target = null;
    TargetSource targetSource = this.advised.getTargetSource();
    try {
        if (this.advised.exposeProxy) {
            // Make invocation available if necessary.
            oldProxy = AopContext.setCurrentProxy(proxy);
            setProxyContext = true;
        }
     // ....
}

又是在拦截方法中处理的,AopContext内部使用ThreadLocal存储的代理对象,所以可以通过AopContext获取到代理对象了。需要说明:此功能需要手动启用,配置如下:

@EnableAspectJAutoProxy(proxyTargetClass = true, exposeProxy = true)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值