【发布时间】: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