前言

之前写过 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. Pandas库学习笔记(1)

    参考:菜鸟教程 pandas库使用了NumPy的大多数功能.建议您先阅读有关NumPy的教程,然后再继续本教程. Pandas 适用于处理以下类型的数据: 与 SQL 或 Excel 表类似的,含异构 ...

  2. Odoo 美化登录界面

    实践环境 Odoo 14.0-20221212 (Community Edition) Odoo Web Login Screen 14.0 https://apps.odoo.com/apps/mo ...

  3. pytest批量执行多个测试文件(批量执行一个文件夹下的所有测试用例)

    图片 代码 #!/usr/bin/env python # @File : test_runall.py import pytest import os # path = os.path.dirnam ...

  4. 解决Win平台VSCode中Python在控制台输出中文乱码的问题

    在菜单Debug->Open Configurations,打开launch.json,新增如下粉红色字符内容: { "configurations": [ { " ...

  5. 七天.NET 8操作SQLite入门到实战 - (3)第七天Blazor学生管理页面编写和接口对接

    前言 本章节的主要内容是完善Blazor学生管理页面的编写和接口对接. 七天.NET 8 操作 SQLite 入门到实战详细教程 第一天 SQLite 简介 第二天 在 Windows 上配置 SQL ...

  6. 关于@Mapper与@MapperScan

    o.m.s.mapper.ClassPathMapperScanner : Skipping MapperFactoryBean with name 'gggDao' and 'com.xxx.dao ...

  7. 【Java】Main方法的命令行参数

    可以使用命令行注入参数执行

  8. 【Mybatis-Plus】01 快速上手

    [官网快速上手地址] https://mp.baomidou.com/guide/quick-start.html#%E5%88%9D%E5%A7%8B%E5%8C%96%E5%B7%A5%E7%A8 ...

  9. Ubuntu18.04环境下安装网络代理软件 proxychains

    安装: 网络代理软件proxychains安装: sudo apt-get install proxychains 为保证使用 proxychains 时 sudo proxychains 时可以实现 ...

  10. 强化学习算法真的适合于你的应用吗 —— 强化学习研究方向(研究领域)现有的不足(短板、无法落地性) —— Why You (Probably) Shouldn’t Use Reinforcement Learning

    外文原文: Why You (Probably) Shouldn't Use Reinforcement Learning 地址: https://towardsdatascience.com/why ...