实体框架核心更新复制

时间:2018-06-29 18:50:07

标签: c# asp.net-core entity-framework-core

当尝试使用子实体集合更新实体时,子集合的每个成员都会重复。为什么在控制器中使用Update方法时子实体会重复?

此应用程序使用ASP.NET Core 2.1和Entity Framework Core,它们连接到Visual Studio 2017中的MS SQL Server LocalDB。

ParentEntity.cs

namespace MyProject.Models
{

    public enum EntityType { Type1, Type2, Type3}

    public class ParentEntity
    {
        public int ParentEntityID { get; set; }

        public string Text { get; set; }

        public EntityType EntityType { get; set; }

        public ICollection<ChildEntity> ChildEntities { get; set; }

    }
}

ChildEntity.cs

namespace MyProject.Models
{
    public class ChildEntity
    {
        public int ChildEntityID { get; set; }
        public string ChildEntityText { get; set; }

        public int ParentEntityId { get; set; }
        public ParentEntity ParentEntity { get; set; }
    }
}

EditEntity.cshtml

@model ParentEntity

<h1>Edit Entity</h1>
<form asp-controller="Entity" asp-action="Edit" method="post">
    <input type="hidden" asp-for="ParentEntityID" />
    <input type="hidden" asp-for="EntityType" />
    <label asp-for="Text"></label>
    <input asp-for="Text" class="form-control" />

    @{ int i = 1; var a = Model.ChildEntities.ToArray(); }
    @for(var c = 0; c < Model.ChildEntities.Count(); c++)
    {
        <label>Child Entity @i</label>
        <input name="ChildEntities[@c].ChildEntityText" value="@a[c].ChildEntityText" />
        i++;
    }

    <button type="submit">Save</button>
</form>

EntityController.cs

[HttpPost]
[ValidateAntiForgeryToken]
public async Task<IActionResult> Edit(int id, [Bind("ParentEntityID, Text, EntityType, ChildEntities")] ParentEntity pa)
{
    if (id != pa.ParentEntityId)
    {
        return NotFound();
    }

    if (ModelState.IsValid)
    {
        try
        {
            context.Update(pa);
            await context.SaveChangesAsync();
        }
        catch (DbUpdateConcurrencyException)
        {

        }
        return RedirectToAction("EditEntity");
    }
    return View(pa);
}

2 个答案:

答案 0 :(得分:1)

我要检查的第一件事是,当从控制器传入ChildEntities上的ChildEntityIds时,是否真的对其进行了设置。如果id为0,EF将认为它们是新实体,并进行插入。

如果它们的ID确实设置为非零整数,我将尝试对它们进行循环并附加实体,以便上下文知道它们。像这样:

foreach(var child in pa.children){
    context.Attach(child);
}

await context.SaveChanges();

答案 1 :(得分:0)

所需要做的就是添加到视图中for循环的每一行的后面。

<input type="hidden" name="ChildEntities[@c].ChildEntityID" value="@a[c].ChildEntityText" class="form-control" />