问题描述:
通过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;