前言

之前写过 EF Core 悲观并发, 这篇主要讲一下乐观并发.

乐观并发的机制可以看这篇.

Why Need This?

如果你用 EF Core 做数据管理, 建议你每个 Entity 都配置乐观并发.

因为 EF 的机制是这样的. 先把资料读出来, 然后修改资料, 然后 SaveChanges.

这个 SaveChanges 并不会把所有的资料都放入 update 语句中, 它会依据之前拿出来的资料做 compare, 如果发现值不同才会生产语句.

由于有了这个"依赖读的资料做逻辑判断" 所以就会有并发的问题了.

比如读出来的数据是 columnA = value1

间中有人把 value1 换成 value2

SaveChange 的时候, 由于我们没有修改 columnA 所以 EF Core 不会生产语句 set columnA = value1

所以哪怕我是在另一个人之后做了 update, 但是数据库这时并不是我以为的 value1. (我们的逻辑告诉我们, entity save changes 之后, 数据库资料 should be 和我的 entity 一样阿)

为了解决这样的混乱, 做一个乐观并发就可以了

主要参考

Handling Concurrency Conflicts

Concurrency Tokens

ConcurrencyToken vs RowVersion

RowVersion vs ConcurrencyToken In EntityFramework/EFCore

Conflict detection in EF Core

RowVersion

在 Entity 加入 RowVersion Property

public class Product
{
public int Id { get; set; }
public string Name { get; set; } = "";
public int Age { get; set; }
public byte[] RowVersion { get; set; } = null!;
}

Setup Model

modelBuilder.Entity<Product>().Property(e => e.RowVersion).IsRowVersion();

执行

var product = db.Products.AsTracking().First();
product.Name = "New Product Name";
try
{
db.SaveChanges();
}
catch (DbUpdateConcurrencyException ex)
{
}

运行的语句多了一个 Where RowVersion = old row version value

如果间中有人对这个 row 做了修改 (不管是谁修改的, 哪怕是 SQL Trigger 改了这个 row, row version 都会自动 update)

那么这个 update 就会失败记入 catch.

题外话: SQL Server 的 RowVersion 使用 binary 维护的, 它不是时间概念, 也不是数据 hash, 它就是一个 running number. 只要 insert/update 语句执行它就会改变 (哪怕没有任何 value change).

IsConcurrencyToken

Entity

public int MyRowVersion { get; set; }

Model

modelBuilder.Entity<Product>().Property(e => e.MyRowVersion).IsConcurrencyToken();

执行

var product = db.Products.AsTracking().First();
product.Name = "New Product Name";
try
{
product.MyRowVersion++; // 手动 update
db.SaveChanges();
}
catch (DbUpdateConcurrencyException ex)
{
}

语句

和 RowVersion 区别

整个机制是一样的, 唯一的区别是 ConcurrencyToken 需要自己手动维护, SQL Server 不会在 insert/update 的时候自动去更新这个 row version 的 column.

另外 EF Core 不鼓励我们用 ConcurrencyToken 哦

当 RowVersion 遇上 Trigger

上面我们有说, 如果数据库有 trigger 的话, 我们在 EF 就无法同时修改 2 个 Entity,

比如我修改了 Entity A 和 Entity B, 然后 SaveChanges

当运行 update A table 的时候 trigger 触发了, 然后它去 update 了 B

导致 B 的 row version 被修改了

然后 EF 跑下一句 update Entity B 语句, 加上 RowVersion = old value 就直接失败了.

解决方法有 2 个思路

1. 不要在一个 SaveChanges 里同时修改 2 个有 Trigger 关系的 Entity, 一个一个修改, 确保拿到最新的 RowVersion, 或者你的顺序要正确.

2. 使用 ConcurrencyToken, 自己决定什么时候需要 update row version, 比如 trigger 的时候不更新. 只有 EF Core SaveChanges 才更新.

目前看 2 个方案都都不理想.

第一个就是要 take care of 这种同时 update 多个 entity 在一个 SaveChanges 里的情况, 简单说就完全不要这样子做, 坚持 1 个 1 个 update, 因为即使现在没有 trigger, 但是也可能后来加入 trigger.

那以后还是有报错的可能性. 总不能等报错才 fix, 也不能加 trigger 的时候全场找这种情况.

第二个也很糟, 因为 trigger 不更新 row version 意味着, 在 EF 如果我们想利用乐观并发就要注意, 不可依赖 Trigger 会更新的 column, 比如冗余数据.

因为它没有 RowVersion 的保护, 要依赖就要使用 transaction. 这个写起来不算难, 但如果 miss 掉的话, 它不会报错. 这个挺危险. 一不小心数据就逻辑错了. 比报错还糟糕.

而且还得另外维护一套自己 update row version 的机制.

追根究底就是 EF 和 Trigger 八字不合, 所以就会遇到很多风水的问题咯.

目前我会选择第 2 个解决方案, 因为 2 个都不好,但是第一个影响的范围会比较大,出现的机率也比较高, 第二个出现的机率比较小, 只是要非常小心, 走一步看一步先吧.

EF Core – 乐观并发的更多相关文章

  1. EntityFramework Core高并发深挖详解,一纸长文,你准备好了吗?

    前言 之前有关EF并发探讨过几次,但是呢,博主感觉还是有问题,为什么会觉得有问题,其实就是理解不够透彻罢了,于是在项目中都是用的存储过程或者SQL语句来实现,利用放假时间好好补补EF Core并发的问 ...

  2. EF|CodeFirst数据并发管理

    在项目开发中,我们有时需要对数据并发请求进行处理.举个简单的例子,比如接单系统中,AB两个客服同时请求处理同一单时,应该只有一单请求是处理成功的,另外一单应当提示客服,此单已经被处理了,不需要再处理. ...

  3. EF Core下利用Mysql进行数据存储在并发访问下的数据同步问题

    小故事 在开始讲这篇文章之前,我们来说一个小故事,纯素虚构(真实的存钱逻辑并非如此) 小刘发工资后,赶忙拿着现金去银行,准备把钱存起来,而与此同时,小刘的老婆刘嫂知道小刘的品性,知道他发工资的日子,也 ...

  4. 笔记-EF Core 并发冲突与令牌

    并发标记并发分悲观并发和乐观并发.悲观并发:比如有两个用户A,B,同时登录系统修改一个文档,如果A先进入修改,则系统会把该文档 锁住,B就没办法打开了,只有等A修改完,完全退出的时候B才能进入修改.乐 ...

  5. EF+MySQL乐观锁控制电商并发下单扣减库存,在高并发下的问题

    下订单减库存的方式 现在,连农村的大姐都会用手机上淘宝购物了,相信电商对大家已经非常熟悉了,如果熟悉电商开发的同学,就知道在买家下单购买商品的时候,是需要扣减库存的,当然有2种扣减库存的方式, 一种是 ...

  6. 悲观并发 乐观并发 Entity Framework Core中的并发处理

    悲观并发策略 A用户发起一个请求   开启了事务 查询到了某一条数据 进行修改     在A提交事务之前 其他人都不能对这条数据进行修改 这种策略最常见的一个问题就是死锁  比如A修改X记录,B修改Y ...

  7. EntityFramework Core技术线路(EF7已经更名为EF Core,并于2016年6月底发布)

    官方文档英文地址:https://github.com/aspnet/EntityFramework/wiki/Roadmap 历经延期和更名,新版本的实体框架终于要和大家见面了,虽然还有点害羞.请大 ...

  8. [转]EntityFramework Core技术线路(EF7已经更名为EF Core,并于2016年6月底发布)

    本文转自:http://www.cnblogs.com/VolcanoCloud/p/5572408.html 官方文档英文地址:https://github.com/aspnet/EntityFra ...

  9. 第四节:EF Core的并发处理

    1.说明 和EF版本的并发处理方案一致,需要知道乐观并发和悲观并发的区别,EF Core只支持乐观并发:监控并发的两种方案:监测单个字段和监测整条数据,DataAnnotations 和 Fluent ...

  10. .NET 云原生架构师训练营(模块二 基础巩固 EF Core 更新和迁移)--学习笔记

    2.4.6 EF Core -- 更新 状态 自动变更检测 不查询删除和更新 并发 状态 Entity State Property State Entity State Added 添加 Uncha ...

随机推荐

  1. 使用ES6中Class实现手写PromiseA+,完美通过官方872条用例

    目录 Promise出现的原因 myPromise的实现要点 myPromise的实现 myPromise - 实现简单的同步 myPromise - 增加异步功能 myPromise - 链式调用( ...

  2. 题解:AT_abc360_c [ABC360C] Move It

    背景 机房大佬掉大分了,乐悲. 题意 给你几个箱子和每个箱子里装有的东西 \(a\) 和对应的重量 \(w\),现在要让每个箱子里都装有一个东西,每次可以移动任意一个箱子中的任意一个东西,代价为它的重 ...

  3. 第三节 JMeter安装及配置

    1.官网地址下载 (1)JDK:https://www.oracle.com/cn/java/technologies/downloads/,下载1.8版本以上的,最好下载最新版本(本次下载本次下载了 ...

  4. freemarker+minio实现页面静态化

    什么是页面静态化? 将原本动态生成的网页内容通过某种形式转化为html并存储在服务器上,当用户请求这些页面时就不需要执行逻辑运算和数据库读 优点: 性能:提高页面加载速度和响应速度,还可以减轻数据库. ...

  5. 用jacoco统计JAVA项目测试代码覆盖率

    一.概述 Jacoco 统计的是全量代码覆盖率.它不仅支持生成单元测试的覆盖率,也支持监控生成接口测试,功能测试的覆盖率. 在新一代精准测试技术流的影响中,各大型单位对覆盖率的追求越来越迫切.作为一款 ...

  6. Jmeter调试取样器

    调试取样器(Debug Sampler),生成一个包含JMeter变量或属性值的样本,并且这些值可以在组件[查看结果树]的响应窗格中看到 组件路径:线程组->右键添加->取样器->D ...

  7. 《最新出炉》系列初窥篇-Python+Playwright自动化测试-61 - 隐藏元素定位与操作

    1.简介 对于前端隐藏元素,一直是自动化定位元素的隐形杀手,让人防不胜防.脚本跑到隐藏元素时位置时报各种各样的错误,可是这种隐藏的下拉菜单又没有办法避免,所以非常头痛,这一篇只为交流隐藏元素自动化定位 ...

  8. ArcGIS for Android入门(Java):初体验

    准备工作 开发工具:Android Studio 环境:jdk 11 (首次接触安卓开发,可能有的地方不太对,还请给位大佬多多指点) 项目搭建 打开Android Studio,点击New Proje ...

  9. NVIDIA公司在实体机器人上的第一步尝试 —— Nova Cater AMR —— 九号机器人与英伟达联合开发的自动驾驶研发平台“Nova Cater AMR(简称:NC)”

    相关: https://www.leiphone.com/category/robot/Hgy9i8azqGncESIB.html Nova Cater AMR是一款仓储运货机器人,可以应用在仓储物流 ...

  10. 大语言模型可以自动生成sql语句吗?

    大语言模型的能力已经是毋庸置疑的了,随着ChatGPT的霸榜,各种语言模型的应用也多了起来,这时候突然有一个意外,那就是:大语言模型可以自动生成sql语句吗? 之所以有这个疑问,主要是因为sql正好是 ...