【问题标题】:Entity Framework Core 3.1.7 : failure during Update of recordEntity Framework Core 3.1.7:更新记录失败
【发布时间】:2020-12-06 23:29:58
【问题描述】:

我有一个最简单的 ASP.NET Core 3.1 Web API 解决方案,模型项目使用 Entity Framework Core 3.1.7 将我的模型映射到 SQL Server。

我有以下实体:

[Table("SessionDetails")]
public class SessionDetailsModel
{
    public long Id { get; set; }
    public DateTime CreateDate { get; set; }
    public Guid Token { get; set; }
    public bool IsValid { get; set; }
}

在每次用户调用登录端点 (POST) 时创建。

db上下文配置如下:

public class RunMeDbContext : DbContext
{
    public RunMeDbContext(DbContextOptions<RunMeDbContext> options) : base(options)
    {
    }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<SessionDetailsModel>()
            .HasKey(x => new {x.Id, x.CreateDate});

        base.OnModelCreating(modelBuilder);
    }

    public DbSet<SessionDetailsModel> SessionDetails { get; set; }
}

插入成功,但是当我调用注销端点时,该端点通过令牌字段获取先前创建的SessionDetails 记录,并尝试将“IsValid”字段(布尔)从真更新为假,我得到以下异常:

Microsoft.EntityFrameworkCore.DbUpdateConcurrencyException:数据库操作预计会影响 1 行,但实际上影响了 0 行。自加载实体以来,数据可能已被修改或删除。有关理解和处理乐观并发异常的信息,请参阅 http://go.microsoft.com/fwlink/?LinkId=527962

我是唯一一个连接到数据库的人,在阅读了并发冲突之后,我可以保证没有对记录进行其他更改。

我的代码:

    public async Task<bool> InvalidateSession(Guid token)
    {
        var session = await _dbContext.SessionDetails.FirstOrDefaultAsync(sd => sd.Token.Equals(token))
            .ConfigureAwait(false);

        if (session == null)
            return false;

        session.IsValid = false;
        
        _dbContext.Update(session);
        await _dbContext.SaveChangesAsync().ConfigureAwait(false);

        return true;
    }

提示:调试时我注意到虽然我要更改的唯一字段是“IsValid = false”,但出于某种奇怪的原因,Token 字段也被修改了(尽管它的值相同):

正如您在此屏幕截图中看到的:

【问题讨论】:

  • 可疑的详细信息会调用Update(),这对于跟踪的实体是不必要的(只需加载实体,设置状态,SaveChanges)使用Single 明确查询是有益的其中需要 1 条记录,而不是 FirstOrDefault。对于像这样的快速操作,我也会避免异步/等待,而只使用同步操作。异步更适合需要一段时间才能完成的操作,它只是用于所有内容时的性能/调试开销。

标签: c# entity-framework entity-framework-core .net-core-3.1 ef-core-3.1


【解决方案1】:

我无法重现这种行为。看看你是否可以修改它以展示你正在观察的行为。

using Microsoft.Data.SqlClient;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Logging;
using System;
using System.ComponentModel.DataAnnotations.Schema;
using System.Linq;

namespace EfCore3Test
{

    [Table("SessionDetails")]
    public class SessionDetailsModel
    {
        public long Id { get; set; }
        public DateTime CreateDate { get; set; }
        public Guid Token { get; set; }
        public bool IsValid { get; set; }
    }
    public class Db : DbContext
    {
           
        public Db(): base()
        {

        }

        private static readonly ILoggerFactory loggerFactory = LoggerFactory.Create(builder =>
        {
            builder.AddFilter((category, level) =>
               category == DbLoggerCategory.Database.Command.Name
               && level == LogLevel.Information).AddConsole();
        });


        protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
        {
            var constr = "Server = localhost; database = efcore3test; integrated security = true";

            optionsBuilder.UseLoggerFactory(loggerFactory)
                          .UseSqlServer(constr, o => o.UseRelationalNulls());

            //optionsBuilder.UseLazyLoadingProxies();
            base.OnConfiguring(optionsBuilder);
        }

        public DbSet<SessionDetailsModel> SessionDetails { get; set; }

        protected override void OnModelCreating(ModelBuilder modelBuilder)
        {
            modelBuilder.Entity<SessionDetailsModel>()
              .HasKey(x => new { x.Id, x.CreateDate });

            modelBuilder.Entity<SessionDetailsModel>().Property(e => e.Id).UseIdentityColumn();

            modelBuilder.Entity<SessionDetailsModel>().HasIndex(e => e.Token).IsUnique(true);

            base.OnModelCreating(modelBuilder);            
        }
    }


    class Program
    {
        static void Main(string[] args)
        {

            long id;
            DateTime created = DateTime.Now;
            Guid token = Guid.NewGuid();
            
            using (var db = new Db())
            {

                db.Database.EnsureDeleted();
                db.Database.EnsureCreated();

                var s = new SessionDetailsModel();
                s.CreateDate = created;
                s.Token = token;
                s.IsValid = true;
                db.SessionDetails.Add(s);
                db.SaveChanges();
                id = s.Id;
            }
            using (var db = new Db())
            {
                var sd = db.SessionDetails.Where(d => d.Token == token).Single();
                sd.IsValid = false;
                db.SaveChanges();
            }
        }
    }

}

【讨论】:

    猜你喜欢
    • 2018-03-21
    • 1970-01-01
    • 2013-02-26
    • 1970-01-01
    • 2016-12-20
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多