Spring Boot-Transactional不进行回滚

问题描述:

通过AOP实现声明式事务–发现只有用户业务逻辑层(UserInfoServiceImpl)未执行回滚操作。

原因分析:

@Override
    public void updateUser(SysUser sysUser) {

        sysUser.setUpdateBy(SecurityUtils.getLoginUser().getId())
                .setUpdateTime(DateUtils.parseDateToString("yyyy-MM-dd HH:mm:ss",new Date()));

        //删除用户角色
        userInfoMapper.delUserRole(sysUser.getId());

        //生成用户角色关系
        List<SysUserRole> sysUserRoleList = creatUserRole(sysUser);

        userInfoMapper.updateUser(sysUser);
        userInfoMapper.insertUserRole(sysUserRoleList);

    }

1、从代码中可以看出我是先删除用户原有角色再封装用户新角色最后再执行用户信息更新操作。此时我将执行用户信息更新sql写错以便验证是否可以进行事务回滚

2、验证结果不尽人意,未进行事务回滚,然后进行排查:

2.1 aop实现声明式事物配置中扫描的包路径没问题

2.2 方法可见度为public没问题

2.3 方法中也未进行try - catch操作

2.4 数据库表引擎为InnoDB

2.5 其他实体的业务逻辑层可以进行回滚

综合以上排查结果那么问题到底出在哪里呢?

3、最后进行了一系列百度后得知:我的框架集成shiro的原因:

ShiroFilterFactoryBean实现了FactoryBean,而且Spring也会加载它的依赖->登录验证(AuthorizingRealm)而实现登陆验证则会依赖UserInfoServiceImpl这个bean。当spring处理UserInfoServiceImpl这个Bean的AOP相关逻辑时发现这个bean已经被优先加载了,就不会再对其进行加载。导致于UserInfoServiceImpl实现不了事务回滚

解决方案:

在AuthorizingRealm里面的UserInfoServiceImpl上面加入注解@Lazy,使其加载顺序不跟着ShiroFilterFactoryBean。

 @Autowired
    @Lazy//使UserInfoService这个类加载于AnnotationAwareAspectJAutoProxyCreator(原因是不加载于(注解生效处理器)AnnotationAwareAspectJAutoProxyCreator会导致事物不回滚)
    private UserInfoService userInfoService;
Spring Boot中,要确保事务能够正常回滚,可以遵循以下几个策略: 1. **公共方法和可见性**: - 使用`@Transactional`注解时,确保该注解应用到public或受保护(protected)的方法上,因为非公开的方法默认会开启事务[^1]。 2. **try-catch语句与异常处理**: - 避免在@Transactional方法内直接捕获并处理异常,因为这可能导致事务回滚。如果需要处理特定异常,应在finally块中完成资源清理。 3. **跨线程事务管理**: - 如果在多线程环境中使用@Transactional,考虑使用Spring的PlatformTransactionManager,如JtaTransactionManager或者设置PROPAGATION_REQUIRED以确保在一个共享的事务上下文中执行[^2]。 4. **检查异常和运行时异常**: - 检查异常(CheckedException)会中断事务流程,而运行时异常(RuntimeException)则会。避免抛出CheckedException,除非确实需要事务回滚。 5. **数据库支持**: - 确保使用的数据库支持事务,某些NoSQL数据库可能支持传统的ACID特性,需要查阅文档确认其事务处理机制。 示例代码: ```java @Service @Transactional(propagation = Propagation.REQUIRED) public class MyService { public void myMethod() { try { // 业务逻辑 throw new RuntimeException("故意引发异常"); } catch (RuntimeException e) { // 异常处理,但要在这里直接回滚,让Spring管理事务 log.error("Error", e); } } } ``` 在上述代码中,如果`myMethod`中的`throw new RuntimeException`被执行,事务会被回滚
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值