0

0

c# DbUpdateConcurrencyException 的处理策略

月夜之吻

月夜之吻

发布时间:2026-01-11 08:30:02

|

318人浏览过

|

来源于php中文网

原创

DbUpdateConcurrencyException 是 EF Core 的乐观并发异常,非数据库锁失败,而是因并发令牌(如[Timestamp])检测到实体被其他操作修改后 SaveChanges() 时触发;需手动配置令牌、模拟冲突验证,并通过原始值/数据库值/当前值三者比对实现安全重试。

c# dbupdateconcurrencyexception 的处理策略

DbUpdateConcurrencyException 是什么,为什么它不等于“数据库冲突”

这个异常不是数据库层面的锁失败或死锁报错,而是 EF Core 在执行 SaveChanges() 时,发现当前要更新的行在加载之后已被其他操作修改过——它依赖的是你配置的并发令牌(concurrency token),比如 [Timestamp][ConcurrencyCheck] 标记的属性。没配令牌?那它根本不会抛这个异常。

如何触发并验证并发检测是否生效

最直接的方式是手动模拟:查出实体 → 另一个上下文改数据库同一行 → 当前上下文调用 SaveChanges()。注意两点:

  • 必须启用并发令牌,例如在实体中加 [Timestamp] 字段(类型为 byte[])或用 Fluent API 配置 IsConcurrencyToken()
  • EF Core 默认只对被标记为 Modified 的属性做 WHERE 条件,所以如果你没改任何字段但只调 Update(),它仍会带旧的并发值去比对
modelBuilder.Entity()
    .Property(e => e.RowVersion)
    .IsRowVersion()
    .IsConcurrencyToken();

捕获后怎么重试才不丢数据

不能简单地 context.Entry(entity).Reload() 再改一遍就 Save,因为用户可能已基于旧状态做了多步逻辑判断(比如库存校验、金额计算)。推荐用“客户端合并”策略:

jQuery自适应宽度跟高度相册代码
jQuery自适应宽度跟高度相册代码

jQuery自适应宽度跟高度相册代码是一款带左右箭头,缩略小图切换的jQuery相册代码。本作品由【站长素材】收集整理,转载请注明出处!

下载
  • 从异常中提取原始值(ex.Entries[0].OriginalValues)、数据库当前值(entry.GetDatabaseValues())和当前修改值(entry.CurrentValues
  • 对比差异,决定是覆盖(强制提交)、放弃(回滚业务逻辑)、还是提示用户冲突(如弹窗显示两版字段差异)
  • 若自动合并,记得手动设置 entry.OriginalValues 为数据库最新值,否则下次 Save 还会撞上同一个异常
catch (DbUpdateConcurrencyException ex)
{
    foreach (var entry in ex.Entries)
    {
        var databaseValues = entry.GetDatabaseValues();
        if (databaseValues == null)
        {
            throw new InvalidOperationException("数据库中已无此记录");
        }

        var clientValues = entry.CurrentValues.Clone();
        entry.OriginalValues.SetValues(databaseValues); // 关键:更新 Original 值
        entry.CurrentValues.SetValues(clientValues);     // 恢复用户修改
    }
    context.SaveChanges(); // 重试
}

高并发场景下,乐观锁不是万能的

它适合读多写少、冲突概率低的业务(如用户资料编辑)。但如果像秒杀扣库存这种高频写场景,靠重试 + Reload 容易形成“重试风暴”,响应延迟飙升。这时该考虑:

  • 把关键操作下沉到存储过程 + 数据库行锁(如 UPDATE ... WHERE stock >= @need
  • 用 Redis 做预占(decrement + expire),再异步落库,把并发压力从 SQL Server 转移到缓存层
  • 避免在长事务里 hold 实体对象,尽早调用 AsNoTracking() 查询只读数据

真正难处理的从来不是异常本身,而是业务语义上“谁的修改该被保留”。DbUpdateConcurrencyException 只是把你回避不了的决策点,提前抛到了代码里。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

676

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

320

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

346

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1094

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

357

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

675

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

571

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

414

2024.04.29

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

80

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 6.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号