添加记录使用实体框架复制其他对象

时间:2013-07-31 20:29:13

标签: asp.net-mvc entity-framework entity-framework-5 code-first dbcontext

我正在尝试使用Entity框架在MVC控制器方法中添加新记录。 当我刚使用“InsertOrUpdate”时,audittype重复了。基于Entity Framework adding record with a related object 的回答,我希望能解决这个问题。这是我现在的代码:

控制器:

if (ModelState.IsValid)
{
    Audit newAudit = Factory.GetNew();
    newAudit.Name = model.Name;
    newAudit.Deadline = model.Deadline;
    newAudit.AuditType = auditTypeRepository.Find(model.SelectedAuditTypeId);

    Repository.InsertOrUpdate(newAudit);
    Repository.Save();

    return RedirectToAction(MVC.Audits.Details(newAudit.Id)); 
}

存储库:

public override void InsertOrUpdate(Qdsa.WebApplications.AuditMaster.Data.Audit model)
{
    if (model.Id == default(int))
    {
        // New entity
        context.Audits.Add(model);
    }
    else
    {
        // Existing entity
        model.ModifiedOn = DateTime.Now;
        context.Entry(model).State = EntityState.Modified;
    }
    //If I leave out the code below the AuditType will be duplicated
    if (model.AuditType != null)
    {
        context.Entry<AuditType>(model.AuditType).State = EntityState.Unchanged;
    }
}

public virtual void Save()
{
    context.SaveChanges();
}

所以我认为我解决了这个问题。但是,AuditType也有Child对象。现在这些子对象得到了重复。 添加具有已存在的子对象的实体的正确方法是什么? 因为AuditType是必需的,所以我先不能保存它,然后再更新它。有什么建议吗?

更新: AuditRepostory和AuditTypeRepository都从BaseRepository继承,其上下文为:

protected DBContext context = new DBContext ();

public virtual T Find(int id)
{
    return All.SingleOrDefault(s => s.Id == id);
}

2 个答案:

答案 0 :(得分:1)

我可以想象出这个问题的两个原因:

  • auditTypeRepository.Find执行无跟踪查询(使用.AsNoTracking()
  • 或者您正在为每个存储库使用一个上下文实例,因此RepositoryauditTypeRepository正在处理两个不同的上下文,这确实会导致重复AuditType,因为您不知道t将其附加到与Repository对应的上下文中(除了评论行之外)。

如果是后者,您应该重新考虑您的设计,并将一个上下文实例注入所有存储库,而不是在存储库中创建它。

答案 1 :(得分:0)

我认为问题出在这里:

newAudit.AuditType = auditTypeRepository.Find(model.SelectedAuditTypeId);

改变这样:

newAudit.AuditTypeId = model.SelectedAuditTypeId;