如何在 EF Core 中使用乐观并发控制

简介: 如何在 EF Core 中使用乐观并发控制

什么是乐观并发控制?

乐观并发控制是一种处理并发访问的数据的方法,它基于一种乐观的假设,即认为并发访问的数据冲突的概率很低。在乐观并发控制中,系统不会立即对并发访问的数据进行加锁,而是在数据被修改时,再检查是否有其他并发操作已经修改了数据。如果检测到冲突,系统 再采取相应的措施来解决冲突。

EF Core 内置了使用并发令牌列实现的乐观并发控制,所谓的并发令牌列通常就是被并发操作影响的列。请看本文是如何在 EF Core 中使用乐观并发控制的……

使用步骤

  1. 创建一个 Asp.net console 项目,并从 Nuget 引用 EF 相关的包

Microsoft.EntityFrameworkCore.SqlServer

Microsoft.EntityFrameworkCore.Tools

2.配置并发冲突列

using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
class HouseConfig : IEntityTypeConfiguration<House>
{
  public void Configure(EntityTypeBuilder<House> builder)
  {
    builder.ToTable("T_Houses");
    builder.Property(p => p.Name).IsUnicode().IsRequired();
    // 把 Owner 列配置为并发令牌
    builder.Property(p => p.Owner).IsConcurrencyToken();
  }
}

3.在 Program.cs 编写以下代码:

using Microsoft.EntityFrameworkCore;
Console.WriteLine("请输入您的姓名");
string name = Console.ReadLine()!;
using TestDbContext ctx = new TestDbContext();
// 1.获取数据
var h1 = await ctx.Houses.SingleAsync(h => h.Id == 1);
if (string.IsNullOrEmpty(h1.Owner))
{
  // 2.延迟5秒,方便测试
  await Task.Delay(5000);
  // 3.更新数据
  h1.Owner = name;
  try
  {
    await ctx.SaveChangesAsync();
    Console.WriteLine("抢到手了");
  }
  catch(DbUpdateConcurrencyException ex)
  {
    // 4. 捕捉和处理并发冲突
    var entry = ex.Entries.First();
    var dbValues = await entry.GetDatabaseValuesAsync();
    string newOwner = dbValues.GetValue<string>(nameof(House.Owner));
    Console.WriteLine($"并发冲突,被{newOwner}提前抢走了");
  }
}
// 5.处理数据已存在情况
else
{
  if (h1.Owner == name)
  {
    Console.WriteLine("这个房子已经是你的了,不用抢");
  }
  else
  {
    Console.WriteLine($"这个房子已经被{h1.Owner}抢走了");
  }
}
Console.ReadLine();

4.测试

  1. 清理 T_Houses 表数据,让 Owner 列等于 null
  2. 同时运行两个控制台程序
  3. 在第一个控制台程序输入 Tom 并运行
  4. 在第二个控制台程序输入 Jim 并运行
  5. 第一个控制台返回消息:抢到手了
  6. 第二个控制台则返回消息:并发冲突,被Tom提前抢走了

扩展

  1. 通常可以通过把并发修改的属性设置为并发令牌的方式启用乐观并发控制。
  2. 有时候无法确定到底哪个属性适合作为并发令牌,比如程序在不同的情况下会更新不同的列或者程序会更新多个列,在这种情况下,可以使用设置一个额外的并发令牌属性的方式来使用乐观并发控制。
  3. 如果使用Microsoft SQL Server数据库,可以用一个byte[]类型的属性作为并发令牌属性,然后使用IsRowVersion把这个属性设置为RowVersion类型,这个属性对应的数据库列就会被设置为ROWVERSION类型。对于ROWVERSION类型的列,在每次插入或更新行时,Microsoft SQL Server会自动为这一行的ROWVERSION类型的列生成新值。
1. 增加一个额外的byte[]类型的属性
class House
{
   public long Id { get; set; }
   public string Name { get; set; }
   public string? Owner { get; set; }
   public byte[] RowVer { get; set; }
}
2. 配置并发令牌
builder.ToTable("T_Houses");
builder.Property(h => h.Name).IsUnicode();
builder.Property(h => h.RowVer).IsRowVersion();
3. Update 语句中的 Where 中使用 RowVer 列

4.其它数据库也可以使用 Guid 作为并发令牌控制

5.乐观并发控制能够避免悲观锁带来的性能下降、死锁等问题,推荐使用乐观并发控制而不是悲观锁


目录
打赏
0
0
0
0
37
分享
相关文章
MySQL MVCC全面解读:掌握并发控制的核心机制
【10月更文挑战第15天】 在数据库管理系统中,MySQL的InnoDB存储引擎采用了一种称为MVCC(Multi-Version Concurrency Control,多版本并发控制)的技术来处理事务的并发访问。MVCC不仅提高了数据库的并发性能,还保证了事务的隔离性。本文将深入探讨MySQL中的MVCC机制,为你在面试中遇到的相关问题提供全面的解答。
550 2
MySQL事务原理分析(ACID特性、隔离级别、锁、MVCC、并发读异常、并发死锁以及如何避免死锁)
MySQL事务原理分析(ACID特性、隔离级别、锁、MVCC、并发读异常、并发死锁以及如何避免死锁)
257 1
【Java|多线程与高并发】CAS以及ABA问题
CAS(Compare and Swap,“比较和交换”)是一种并发编程中常用的原子操作,用于解决多线程环境下的数据竞争和并发访问问题。
精通Java事务编程(7)-可串行化隔离级别之两阶段锁定(2PL,two-phase locking)
近30年,DB只有一种广泛使用的串行化算法:两阶段加锁 1 2PL不是2PC 请注意,虽然两阶段锁定(2PL)听起来非常类似于两阶段提交(2PC),但是完全不同概念
268 0
【MySQL】MVCC多版本并发控制(重点:MVCC实现原理之ReadView)
本文重点介绍MySQL的MVCC概念、快照读与当前读、MVCC实现原理之ReadView、隐藏字段、Undo Log版本链。
570 0
【Java原理探索】「并发原理专题」AQS的技术体系之CLH、MCS锁的原理及实现
【Java原理探索】「并发原理专题」AQS的技术体系之CLH、MCS锁的原理及实现
155 0
【Java原理探索】「并发原理专题」AQS的技术体系之CLH、MCS锁的原理及实现
事务并发控制技术
事务并发控制技术
171 0
事务并发控制技术
Java并发基石CAS原理以及ABA问题
在学习CAS之前,先从一个简单的案例入手,进而引出CAS的基本使用
Java并发基石CAS原理以及ABA问题
EF 事务(非分布式事务)
在EF 中怎么使用事务? 这个问题纠结了我好久,直到有人跟我一起讨论,我和同事一起讨论查资料。 查的好多资料都是使用 TransactionScope,用 TransactionScope 可处理分布式事务。
1485 0
锁机制有什么用?简述Hibernate的悲观锁和乐观锁机制
有些业务逻辑在执行过程中要求对数据进行排他性的访问,于是需要通过一些机制保证在此过程中数据被锁住不会被外界修改,这就是所谓的锁机制。 Hibernate支持悲观锁和乐观锁两种锁机制。
1659 0
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等