Spring事务失效常见的八种场景
admin
2024-02-04 06:11:35
0

文章目录

    • 1. 方法自调用
    • 2. 方法修饰符为private
    • 3. 方法是final的
    • 4. 单独的线程调用
    • 5. Spring中没加@Configuration注解
    • 6. 异常被吃掉
    • 7. 类没有被Spring管理
    • 8. 数据库没有开启事物

1. 方法自调用

第一类,@Transactional注解未生效情况,其实这种并不是事物失效,仅仅是注解失效,注解写了更没写一样

解决方案:

  • 将被调用的方法拆到单独的bean中,让切面起作用
  • 自己注入自己获取代理类
  • @EnableAspectJAutoProxy(exposeProxy = true) + UserService userService = (UserService) AopContext.currentProxy();

Spring事务是基于AOP的,只有当代理对象调用某个方法时,Spring的事务才会生效,而在一个方法中调用this.xxx()方法时,由于this并不是代理对象,所以会导致事务失效

@Transactional
public void a(){jdbcTemplate.execute("insert into t1 values(1)");throw new RuntimeException();
}// 自调用
public void b(){// 由于调用者并不是代理对象,所以切面失败导致事物失效a();
}

还有一种情况

// 此时相当于`a`方法上面根本就没贴事物注解一样
@Transactional(propagation = Propagation.NEVER)
public void a(){jdbcTemplate.execute("insert into t1 values(1)");throw new RuntimeException();
}// 调用方法也加了事物注解,此时会进行回滚,但是是由b方法的事物就行回滚的
@Transactional
public void b(){// 由于调用者并不是代理对象,所以切面失败导致事物失效a();
}

所以解决方案就是用代理对象调用即可

public class UserService {@Autowired // 自己注入自己private UserService userService;@Transactionalpublic void a(){jdbcTemplate.execute("insert into t1 values(1)");throw new RuntimeException();}// 事物生效public void b(){userService.a();}
}

或者可以通过代理上下文获取当前类的代理对象,但是这种需要我们在配置类上面设置属性@EnableAspectJAutoProxy(exposeProxy = true)

public class UserService {@Transactionalpublic void a(){jdbcTemplate.execute("insert into t1 values(1)");throw new RuntimeException();}// 事物生效public void b(){// 获取当前类的代理对象UserService userService = (UserService) AopContext.currentProxy();userService.a();}
}

2. 方法修饰符为private

  • cglib代理是基于父子类的,如果方法私有无法在子类中获取到,无法重写,只能执行父类的方法,但是父类中没有jdbcTemplate,会抛出NPE
  • 多态口诀:new谁调用谁的方法,谁创建使用谁的属性
public class UserService {// cglib代理是基于父子类的,如果方法私有无法在子类中获取到,只能执行父类的方法,但是父类中没有`jdbcTemplate`,`NPE`@Transactionalprivate void a(){jdbcTemplate.execute("insert into t1 values(1)");throw new RuntimeException();}
}

3. 方法是final的

同上,代理类无法重写代理的方法

4. 单独的线程调用

Mybatis或者JdbcTemplate执行SQL时,会从ThreadLocal中去获取数据库连接对象,如果开启事物的线程和执行SQL的线程不是同一个线程,就拿不到数据库连接对象,这样,Mybatis或者JdbcTemplate就会重新创建一个数据库连接用来执行SQL,如果这个事物自动事物提交autocommit默认开启的话,就不会因为异常而进行回滚

public class UserService {// 事物失效@Transactionalprivate void a(){new Thread(() -> {jdbcTemplate.execute("insert into t1 values(1)");throw new RuntimeException();}).start();}
}

5. Spring中没加@Configuration注解

在Spring中,由于MybatisJdbcTemplate会从ThreadLocal中获取数据库连接,但是ThreadLocal里面存的是map,即ThreadLocal>的结构

如果没有添加@Configuration注解,会导致Map中存的DateSource和MybatisJdbcTemplate对象不相等,从而拿不到数据库连接,导致自己会新建一个,从而导致事物失效

注意:在SpringBoot中由于自动装箱,不会出现这种问题

6. 异常被吃掉

Spring默认只会对RuntimeExceptionError进行回滚

7. 类没有被Spring管理

8. 数据库没有开启事物

相关内容

热门资讯

从账户受限到税务规划:厦门家庭... 2026年6月,土耳其第7582号法律正式公布,为符合条件的税务居民提供最长20年的境外收入免税期。...
儿科崔雪梅:孩子得了抽动症,家... 家有萌娃,本应是充满欢声笑语的幸福时光,但如果孩子被诊断出患有抽动症,家长们往往会陷入焦虑和迷茫之中...
原创 今... 近900亿美元认购需求,转眼变成4亿美元账面亏损。债市用最残酷的方式告诉马斯克:股票投资者可以陪你去...
原创 稀... 最近国际新闻里有个挺热闹的事,欧盟一位叫西凯拉的委员跑到巴西待了一周,公开喊话要给巴西一个比中国和美...
原创 四... 四十年代位于北京西单旧刑部街的“王光超大夫诊所”,表面是一家私人医疗诊所,实则是中共北平地下党的重要...
原创 “... 俄罗斯前央行顾问表示,尽管战争引发的经济困境并未动摇弗拉基米尔·普京的权力基础,但 这个政权最终走向...
原创 美... 今年2月,阿斯麦宣布要在全球裁掉大约1700个岗位。这本来是一条很正常的商业新闻,但彭博社等美西方媒...
原创 巴... 2026年5月28日,坐落于巴西米纳斯吉拉斯州的科罗苏斯稀土研发加工中心正式落成,运营方为澳大利亚维...
视频号广告服务企业梳理:工业制... 导语:根据《2025-2026中国短视频与直播营销白皮书》及多家第三方机构数据,视频号广告服务市场近...
原创 黄... 黄金这一跌,把两类人同时照了出来:一类是嘴上喊着“回调就买”的围观者,另一类是把客户定金当筹码的黄金...
半导体大牛股,紧急发布澄清声明 【导读】富信科技发布澄清公告 中国基金报记者 莫琳 6月28日,针对部分网络媒体、自媒体及社交媒体平...
脑佳科技CEO蒲云海:在真实场... 封面新闻记者 欧阳宏宇 无需开刀,靠意念就能操控机械手……近年来,脑机接口正逐步走进医疗领域,尤其是...
原创 6... 中国并没有选择让伊朗拖垮美国,这背后的算盘比表面复杂得多。 很多人以为中国是在“帮美国解围”,但其实...
三条全球最大碳纤维产线同日投产... 来源:第一财经 我国碳纤维产业创新发展再添里程碑。 6月28日,中国建材三条世界级高性能碳纤维生产线...
天津名酒回收行业观察:打破单一... 提起名酒回收,绝大多数人的固有认知,还停留在单一回收茅台、五粮液等国产高端白酒的传统模式。 过去很长...
东莞松山湖科技金融集聚区正式开... 6月25日上午,东莞松山湖科技金融集聚区开园仪式在集聚区金融广场举行。活动现场八大合作平台集中揭牌、...
携手华为“鸿图”打造智慧医院“... 在数字化浪潮奔涌的今天,一部手机控制全屋智能已不稀奇,但在关乎生命健康的医院里,让成千上万的医疗设备...
蜜雪冰城的玩法,口子窖能复制吗... 文 | 创业最前线 “麻雀也能喝二两”这句在安徽淮北濉溪县街头巷尾流传的俗语,最近被口子窖写成了一...
必迈体育冲击IPO,实控人、董... 曾任职李宁CEO的张志勇,如今带领旗下跑步运动垂类品牌“必迈”的运营主体北京必迈体育股份有限公司(以...
OpenAI推迟上市,那“Ki... 来源:虎嗅APP AI的估值逻辑变了,“Kimi们”准备好了吗? 出品|虎嗅科技组 作者|宋思杭 ...