c# - EF Core 5 - 更新多对多关系中的实体会引发 "Violation of PRIMARY KEY"错误

标签 c# sql-server asp.net-core entity-framework-core many-to-many

我正在使用 .NET 5并使用 EF Core和用于数据库的 MSSQL

我有 2 个类,它们之间存在多对多关系。那些从模型迁移到数据库。 example of DB tables

迁移:

migrationBuilder.CreateTable(
            name: "TournamentUser",
            columns: table => new
            {
                TournamentsId = table.Column<int>(type: "int", nullable: false),
                UsersId = table.Column<int>(type: "int", nullable: false)
            },
            constraints: table =>
            {
                table.PrimaryKey("PK_TournamentUser", x => new { x.TournamentsId, x.UsersId });
                table.ForeignKey(
                    name: "FK_TournamentUser_Tournaments_TournamentsId",
                    column: x => x.TournamentsId,
                    principalTable: "Tournaments",
                    principalColumn: "Id",
                    onDelete: ReferentialAction.Cascade);
                table.ForeignKey(
                    name: "FK_TournamentUser_Users_UsersId",
                    column: x => x.UsersId,
                    principalTable: "Users",
                    principalColumn: "Id",
                    onDelete: ReferentialAction.Cascade);
            });

模型:

public class Tournament
{
    [Key]
    public int Id { get; set; }
    [Required, MaxLength(20)]
    public string Name { get; set; }
    [Required]
    public int MaxPlayers { get; set; }
    public int PlayersCount { get; set; }
    [Required]
    public DateTime Start { get; set; }
    public bool IsStarted { get; set; }
    public bool IsEnded { get; set; }
    public List<User> Users { get; set; }
}

public class User
{
    [Key]
    public int Id { get; set; }
    [Required, MaxLength(15)]
    public string Username { get; set; }
    [Required]
    public string Password { get; set; }
    [Required, MaxLength(20)]
    public string Name { get; set; }
    [Required, MaxLength(20)]
    public string Surname { get; set; }
    [JsonIgnore]
    public List<Tournament> Tournaments { get; set; }
}

所以当我尝试更新锦标赛模型时

[HttpPut]
public async Task<ActionResult<Tournament>> Put([FromBody] Tournament tournament)
{
    _context.Tournaments.Update(tournament);
    await _context.SaveChangesAsync();

    return CreatedAtAction(nameof(GetTournament), new { id = tournament.Id }, tournament);
}

我收到这个错误

Microsoft.Data.SqlClient.SqlException (0x80131904): Violation of PRIMARY KEY constraint 'PK_TournamentUser'. Cannot insert duplicate key in object 'dbo.TournamentUser'. The duplicate key value is (1, 1).

对此有什么可能的解决方案?我想保持我的多对多表更新,因为每次更新 Tournament 时都可以删除或添加 Tournament's User

编辑:
以下显示锦标赛对象数据 -

Example of updatable tournament object

最佳答案

解决方案:

要更新 Tournament 实体,请使用以下方法 -

_context.Entry(tournament).State = EntityState.Modified;
await _context.SaveChangesAsync();

解释:

  1. 如果实体具有 Id(自动生成的主键)值,则 Update() 方法将实体标记为 Modified,如果它没有 Id 值,则为 Added。如果您使用实体图(具有相关实体的实体)调用该方法,则任何相关实体也是如此。然后在下一个 SaveChanges() 调用中,EF 根据实体是否被标记为 Modified 为实体生成更新和/或插入命令已添加。重点是 - Update() 方法不仅更新实体,它还可以插入新实体

  2. 在您的情况下,查看模型,EF 可以告诉 TournamentUser 处于多对多关系中,并且有一个连接数据库级别的实体 TournamentUser。但是由于您是从外部接收实体图,因此 EF 不会跟踪它们,并且它无法判断此锦标赛的 TournamentUser 实体和相关用户是否已存在于数据库中。

  3. 当您使用实体图、锦标赛、相关用户以及与他们相关的加入实体 (TournamentUser) 调用 Update() 方法时全部成为更新操作的对象。 EF 会尝试创建加入实体,就像创建/插入没有 Id 值的新实体一样。

  4. EF 为 TournamentUser 实体生成插入命令,并为 TournamentUser 实体生成更新命令。但是在数据库级别,由于 TournamentUser 链接已经存在,您会收到 Violation of PRIMARY KEY 错误。

-

_context.Entry(tournament).State = EntityState.Modified;

上面建议明确标记锦标赛为已修改。因此,EF 不会费心查看任何相关实体,也不会推断任何关系,而只会为锦标赛生成更新命令。

关于c# - EF Core 5 - 更新多对多关系中的实体会引发 "Violation of PRIMARY KEY"错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66625740/

相关文章:

c# - 从 ASMX Web 服务捕获自定义异常

c# - 按值查找 Xml 属性,并更改其其他值

c# - 我应该如何教我的 friend C#?

sql-server - 数据库镜像/复制,SQL Server 2005

sql-server - 如何在没有CLR的情况下调试SQL Server 2008中的存储过程?

asp.net-core - 带有 anchor 标记助手的命名路由

c# - 如何在 C# 中读取 Http 响应流两次?

c# - 大量数据的插入性能缓慢 (SQL Server/C#)

Razor 找不到类型或命名空间

c# - 无法访问 session asp.net 核心