我正在使用 .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
。
编辑:
以下显示锦标赛对象数据 -
最佳答案
解决方案:
要更新 Tournament
实体仅,请使用以下方法 -
_context.Entry(tournament).State = EntityState.Modified;
await _context.SaveChangesAsync();
解释:
如果实体具有
Id
(自动生成的主键)值,则Update()
方法将实体标记为Modified
,如果它没有Id
值,则为Added
。如果您使用实体图(具有相关实体的实体)调用该方法,则任何相关实体也是如此。然后在下一个SaveChanges()
调用中,EF 根据实体是否被标记为Modified
或为实体生成更新和/或插入命令已添加
。重点是 -Update()
方法不仅更新实体,它还可以插入新实体。在您的情况下,查看模型,EF 可以告诉
Tournament
和User
处于多对多关系中,并且有一个连接数据库级别的实体TournamentUser
。但是由于您是从外部接收实体图,因此 EF 不会跟踪它们,并且它无法判断此锦标赛的TournamentUser
实体和相关用户是否已存在于数据库中。当您使用实体图、锦标赛、相关用户以及与他们相关的加入实体 (
TournamentUser
) 调用Update()
方法时全部成为更新操作的对象。 EF 会尝试创建加入实体,就像创建/插入没有Id
值的新实体一样。EF 为
TournamentUser
实体生成插入命令,并为Tournament
和User
实体生成更新命令。但是在数据库级别,由于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/