实体框架核心3.1.7:记录更新期间失败

如何解决实体框架核心3.1.7:记录更新期间失败

我有最简单的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)时创建的

通过以下方式配置数据库上下文:

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'字段(布尔)从true更新为false,得到以下异常:

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',但出于某些奇怪的原因,令牌字段也被修改了(尽管它的值是相同的):

如您在此屏幕截图中所见:

enter image description here

解决方法

我无法复制这种行为。看看是否可以修改它以显示您正在观察的行为。

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();
            }
        }
    }

}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-