gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-02 02:19:35 25 4
gpt4 key购买 nike

我正在使用 .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 ofPRIMARY KEY constraint 'PK_TournamentUser'. Cannot insert duplicatekey 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/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com