AddOrUpdate 上的 EF 复合主键违规

EF composite primary key violation on AddOrUpdate

我正在尝试创建 2 个表。一个用于商店,一个用于收银机。收银机有组合键:Id + ShopId

这是一个模型:

public class Shop
{
    [Key]
    public int Id { get; set; }
    public string ShopName { get; set; }
}

public class CashRegister
{
    [Key, Column(Order = 0)]
    public int Id { get; set; }
    public string CashRegisterName { get; set; }

    [ForeignKey("ShopId")]
    public Shop Shop { get; set; }

    [Key, Column(Order = 1)]
    public int ShopId { get; set; }
}

这是迁移的样子:

        CreateTable(
            "dbo.CashRegisters",
            c => new
                {
                    Id = c.Int(nullable: false),
                    ShopId = c.Int(nullable: false),
                    CashRegisterName = c.String(),
                })
            .PrimaryKey(t => new { t.Id, t.ShopId })
            .ForeignKey("dbo.Shops", t => t.ShopId, cascadeDelete: true)
            .Index(t => t.ShopId);

        CreateTable(
            "dbo.Shops",
            c => new
                {
                    Id = c.Int(nullable: false, identity: true),
                    ShopName = c.String(),
                })
            .PrimaryKey(t => t.Id);

下面是抛出主键冲突异常的代码:

        var context = new Model();

        var shops = new List<Shop>
        {
          new Shop() { ShopName = "First shop" },
          new Shop() { ShopName = "Second shop" }
        };

        context.Shops.AddOrUpdate(shops.ToArray());
        context.SaveChanges();

        var cashRegisters = new List<CashRegister>();

        foreach (var shop in shops)
        {
            cashRegisters.Add(new CashRegister()
            {
                CashRegisterName = "First cash register",
                ShopId = shop.Id
            });

            cashRegisters.Add(new CashRegister()
            {
                CashRegisterName = "Second cash register",
                ShopId = shop.Id
            });
        }

        context.CashRegisters.AddOrUpdate(a => new { a.Id, a.ShopId }, cashRegisters.ToArray());
        context.SaveChanges();
        //primary key violation exception on line above

现在显然收银机在创建时将 0 作为 Id。 我得到的例外是:

"Violation of PRIMARY KEY constraint 'PK_dbo.CashRegisters'. Cannot insert duplicate key in object 'dbo.CashRegisters'. The duplicate key value is (0, 1).\r\nThe statement has been terminated."

我尝试将 [DatabaseGenerated(DatabaseGeneratedOption.Identity)] 添加到 CashRegister.Id 但后来我得到另一个异常 {"Cannot insert the value NULL into column 'Id', table 'TestCompositeKeys.dbo.CashRegisters'; column does not allow nulls. INSERT fails.\r\nThe statement has been terminated."},这很奇怪,因为 Id 甚至不能为空。

谁能告诉我我做错了什么或者我该如何解决这个问题?谢谢

此代码在这里

foreach (var shop in shops)
        {
            cashRegisters.Add(new CashRegister()
            {
                CashRegisterName = "First cash register",
                ShopId = shop.Id
            });

            cashRegisters.Add(new CashRegister()
            {
                CashRegisterName = "Second cash register",
                ShopId = shop.Id
            });
        }

将插入4个CashRegister对象。两个商店各两个。当 ShopId 设置为主键时,CashRegister Table 不能有相同的 ShopId 两次。

解决方案是为每个 Shop 对象插入一个 CashRegister 对象,或者从 ShopId 中删除主要约束。

CashRegister.Id 设置为自动递增实际上应该有效。

public class CashRegister
{
    [Key, Column(Order = 0), DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    public int Id { get; set; }
    // ...

要考虑的另一件事是 AddOrUpdate 语句。由于 Id 只有在实际插入时才会被赋值,所以你不能真正通过 Id 比较条目。相反,您可以尝试以下操作:

context.CashRegisters.AddOrUpdate(a => new { a.ShopId, a.CashRegisterName  }, cashRegisters.ToArray());

它只会使用初始化的属性进行比较,并且会真正防止重新插入,而使用 a.Id, a.ShopId 你可能会在第二次执行种子时得到重复项。