C# core 乐观并发控制

Entity Framework (EF) 本身并不提供专门的"金额锁"操作,因为这通常是业务逻辑的一部分,而不是数据访问层面的功能。不过,如果你想确保对数据库中的金额字段进行安全的更新,以避免并发问题,可以使用乐观并发控制或悲观锁定策略。

乐观并发控制

乐观并发控制假设多个事务更新同一记录的冲突概率较低。它允许多个进程同时读取一个记录,并在提交更新时检查记录是否自上次读取以来被修改过。如果记录已被修改,则更新会失败。在Entity Framework中,这通常是通过在模型中使用ConcurrencyToken来实现的。

例如,在模型定义中,你可以指定一个属性作为并发令牌:

csharp 复制代码
csharp复制
public class Account
{
    public int AccountId { get; set; }
    [ConcurrencyCheck]
    public decimal Balance { get; set; }
    // 其他属性...
}

或者在OnModelCreating中使用Fluent API配置:

scss 复制代码
csharp复制
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
    modelBuilder.Entity<Account>()
        .Property(a => a.Balance)
        .IsConcurrencyToken();
}

这样,在更新操作中,EF Core会检查在读取记录后该记录是否被修改过。如果在更新时检测到并发冲突(即记录已被其他事务修改),EF Core会抛出一个DbUpdateConcurrencyException异常。

悲观并发控制

悲观并发控制假设冲突很可能会发生,并且它通过在事务开始时锁定记录来防止冲突。在EF Core中,你可以通过使用事务并显式指定锁定行为来实现悲观锁定,例如使用SELECT ... FOR UPDATE语句。

在EF Core中执行悲观锁定的一个例子:

csharp 复制代码
csharp复制
using (var context = new YourDbContext())
{
    using (var transaction = context.Database.BeginTransaction())
    {
        try
        {
            // 查询要更新的账户,并锁定该记录
            var account = context.Accounts
                .FromSqlRaw("SELECT * FROM Accounts WHERE AccountId = {0} FOR UPDATE", accountId)
                .SingleOrDefault();

            if (account != null)
            {
                // 执行金额更新操作
                account.Balance += amount;  

                context.SaveChanges();
                transaction.Commit();
            }
        }
        catch (Exception)
        {
            transaction.Rollback();
            // 处理异常
        }
    }
}

请注意,不是所有的数据库系统都支持SELECT ... FOR UPDATE(这是SQL悲观锁的一种实现),并且确切的SQL语法可能因数据库类型而异。上面的代码适用于支持此类查询的数据库(如PostgreSQL,MySQL等)。对于SQL Server,你可能会使用不同的锁定语句,例如WITH (UPDLOCK, ROWLOCK)

注意: 使用悲观锁定策略时,要特别注意死锁的可能性以及对性能的影响。

在进行金额更新等重要业务操作时,要确保你的代码可以正确处理并发场景,确保数据的一致性和完整性。如果需要,也应该在事务中执行这些操作,以确保操作的原子性。

相关推荐
Kookoos43 分钟前
性能剖析:在 ABP 框架中集成 MiniProfiler 实现性能可视化诊断
后端·c#·.net·abp vnext·miniprofiler
zhanshuo1 小时前
5分钟手把手实战:用HTML5基础结构打造你的个人简介页面
.net
界面开发小八哥1 小时前
界面开发框架DevExpress XAF实践:集成.NET Aspire后如何实现数据库依赖?
ui·.net·界面控件·devexpress·ui开发·xaf
zhanshuo1 小时前
5分钟搞定!ASP.NET正则表达式验证控件实战:轻松拦截99%的无效邮箱!
.net
阿翰3 小时前
自动 GitHub Readme 20 种语言翻译平台 - OpenAiTx 开源免费
c#·.net
津津有味道1 天前
VB.net复制Ntag213卡写入UID
.net·nfc·ntag213·写uid
lv_fu1 天前
调用.net DLL让CANoe自动识别串口号
.net·dll·capl·canoe·vid pid·自动串口
步、步、为营1 天前
.NET 事件模式举例介绍
java·开发语言·.net
追逐时光者1 天前
免费且全面的C#/.NET/.NET Core面试宝典,阅读量突破40万+了!
后端·.net
qq_297908011 天前
C#报价系统陈列展示成本核算系统项目管理系统纸品非纸品报价软件
sqlserver·c#·.net·开源软件