尧图网站建设 尧图网络
  • 首页
  • 关于我们
  • 服务项目
  • 案例展示
  • 建站流程
  • 资讯中心
  • 联系我们
首页/资讯中心/详情

蓝凌EKP产品:一次 Hibernate 乐观锁 + 死锁的深度踩坑实录

蓝凌EKP产品:一次 Hibernate 乐观锁 + 死锁的深度踩坑实录
📅 发布时间:2026/6/22 10:08:19

—— clear() 一个集合,为什么引发 OptimisticLockException 和数据库死锁?

这是一次看似“新增 / 查询”的普通业务操作,却最终演变成
Hibernate 乐观锁异常 + MySQL 死锁 + 批量更新失败的连环事故。

一、问题现象

线上频繁出现如下异常:

javax.persistence.OptimisticLockException: Batch update returned unexpected row count from update [0]; actual row count: 0; expected: 1

伴随的数据库日志还有:

Deadlock found when trying to get lock; try restarting transaction SQL: delete from km_agreement_review_areader where fd_source_id=?

诡异点在于:

  • 表面调用的是新增审计意见 / 新增参数

  • 堆栈中指向的是find()查询

  • 实际报错却是update / delete

  • 涉及的表包括:

    • km_agreement_review_areader

    • lbpm_execution

怎么看都不像是一个地方的问题。

二、第一个误区:find / insert 不会触发乐观锁?

这是一个非常常见的误区。

在 Hibernate / JPA 中,只要触发了 flush,
所有被托管(managed)的实体都会被统一同步到数据库。

也就是说:

insert A find B

在 Hibernate 内部真实执行顺序是:

flush Session ├─ update / delete / insert(之前积攒的) └─ 然后才是 find B

异常只是“挂”在 find 上,
真正出问题的是 flush 里的 update / delete。

三、真正的导火索:BaseAuthModel 里的clear()

所有业务 Model 都继承了一个公共父类:

public abstract class BaseAuthModel { protected List authAllReaders; protected void recalculateReaderField() { if (authAllReaders == null) { authAllReaders = new ArrayList(); } else { authAllReaders.clear(); // 关键代码 } // 重新 add 各种 reader authAllReaders.add(getDocCreator()); ... } }

这一行代码,看起来极其“正常”:

authAllReaders.clear();

但在Hibernate 眼里,这是一个非常危险的操作。

四、Hibernate 视角:clear() 到底干了什么?

假设映射关系是:

@OneToMany @JoinColumn(name = "fd_source_id") private List<Reader> authAllReaders;

那么:

authAllReaders.clear();

在 Hibernate 中等价于:

delete from km_agreement_review_areader where fd_source_id = ?

如果接下来你又:

authAllReaders.add(...) authAllReaders.add(...)

Hibernate flush 时会执行:

delete from km_agreement_review_areader where fd_source_id = ? insert into km_agreement_review_areader ... insert into km_agreement_review_areader ...

五、为什么会死锁?

关键点:fd_source_id是外键

当多个并发请求同时操作同一个业务对象时:

  • 线程 A:clear → delete(持有子表行锁)

  • 线程 B:clear → delete(等待)

  • 线程 A:后续 update 父表 / execution

  • 线程 B:持有另一批锁

👉典型的 InnoDB 死锁模型

DELETE 子表 → UPDATE 父表 → DELETE 子表(另一个事务)

六、为什么又会触发 OptimisticLockException?

系统中只有一个地方配置了 version:

<version name="lockerVersion" column="fd_locker_version" type="long"/>

配置在:lbpm_execution表。

而问题在于:

  • lbpm_execution在流程执行时早已被加载进 Session

  • clear / add Reader 的过程中:

    • execution 被标记为 dirty

  • flush 时 Hibernate 会顺带执行:

update lbpm_execution set fd_locker_version = fd_locker_version + 1 where fd_id = ? and fd_locker_version = ?

如果前面发生了:

  • 死锁回滚

  • 并发已更新

  • 行被别的事务影响

👉 update 0 行
👉 Hibernate 判定并发冲突
👉 抛OptimisticLockException

七、为什么堆栈看起来“很乱”?

因为这是统一 flush 导致的错觉:

  • 报错点挂在find()

  • 实际执行的是:

    • delete 子表

    • update execution

  • Hibernate不会告诉你是哪一个实体被判定 dirty

这也是 Hibernate 最反直觉的地方之一。

八、问题的本质总结(一句话)

这是一个典型的:
ORM 管理集合 + clear + 并发 + version
共同作用下的“架构级事故”

九、解法一(最推荐):绕过 ORM 管理这个集合

❌ 错误姿势(当前做法)

@OneToMany private List authAllReaders;

并对其:

clear() + add()

✅ 正确姿势 1:不做批量异常在新增,只做增量

这里有现成方案,可以关注后续.....,
public void update(IBaseModel modelObj) throws Exception { IFieldsCalculator fieldsRecalculator = modelObj.getFieldsCalculator(); if(fieldsRecalculator!=null){ fieldsRecalculator.recalculateFields(); }else{ modelObj.recalculateFields(); } //modelObj.recalculateFields(); afterRecalculateFields(modelObj); getHibernateTemplate().saveOrUpdate(modelObj); }

model 实现接口IFieldsCalculator fieldsRecalculator = modelObj.getFieldsCalculator();完成model 的可阅读者的增量更新。

十、为什么这是“无解但必须改”的问题?

因为:

  • BaseAuthModel 是全局父类

  • clear() 是高频路径

  • 并发下必然出现:

    • 死锁

    • 乐观锁冲突

  • Hibernate 在这里帮不了你

👉必须从 ORM 设计上规避

十一、最终总结

Hibernate 非常适合“领域模型”
但需谨慎操作“权限计算 / 批量 clear + 重建”这种模型

这次问题的真正收获不是“修一个 bug”,而是:

  • 明确了哪些集合不该交给 ORM 管

  • 明确了clear() 是 ORM 世界的核按钮

  • 明确了version 不该用在流程执行上下文上

相关新闻

  • Excalidraw负载均衡配置:高并发场景下的稳定性保障
  • 基于Java+SpringBoot+SSM音乐推荐系统(源码+LW+调试文档+讲解等)/音乐推荐算法/音乐智能推荐/歌曲推荐系统/音乐个性化推荐/音乐播放推荐
  • springboot高校应届毕业生求职招聘系统vue_12wlz

最新新闻

  • 湖南音响改装难题终结者:天宇汽车音响连锁(长沙旗舰店)的核心优势,宝马音响改装,音响改装官方门店找哪家 - 音响改装门店分享
  • Claude Code 里那个 Extended Thinking 输出:它到底是什么,以及为什么你不能拿它当审计日志
  • LoRA合并新突破:Pico算法校准输出空间共享方向,提升多任务性能
  • LLM 推理性能优化:从显存管理到推理加速的全链路方案
  • 实战!用Python爬取海关总署进出口贸易数据 —— 从反爬突破到数据可视化全流程指南
  • TRAE SOLO模式:模型无关的AI编程指令抽象层

日新闻

  • Arduino-ESP32项目深度解析:解锁隐藏芯片支持与架构演进
  • 2026年 系统窗厂家/品牌推荐榜单:隔音系统窗+高端系统门窗的核心优势与选购指南 - 品牌发掘
  • NVBench:首个双语非言语发声语音合成评测基准详解与实践

周新闻

  • Visual C++运行库修复终极指南:5分钟快速解决Windows软件启动错误
  • 手把手教你构建统计局地区经济数据爬虫:从环境搭建到数据持久化全指南
  • 2026多Agent深度解析:用AI团队替代单一模型,四种架构实战落地

月新闻

  • 【总结】入门篇:50句话让你记住架构核心概念
  • WeChatMsg技术方案解析:实现Mac微信数据自主管理的完整解决方案
  • WeChatMsg:革新性微信数据备份方案,打造你的专属数字记忆库

关于尧图

  • 公司简介
  • 团队介绍
  • 企业文化
  • 荣誉资质

服务项目

  • 定制开发
  • 电商建站
  • UI 设计
  • 运维服务

快速链接

  • 案例展示
  • 建站流程
  • 常见问题
  • 资讯中心

联系方式

  • 📍北京市朝阳区互联网产业园 A 座 10 层
  • 📞400-888-8888
  • ✉️contact@rkmt.cn
  • 🕐周一至周日 9:00-21:00

© 2024 北京尧图网络科技有限公司 版权所有 | 京 ICP 备 XXXXXXXX 号