EF 4.1:从集合中删除子对象不会删除它 - 为什么?

时间:2011-05-25 14:16:33

标签: c# .net entity-framework poco entity-framework-4.1

我有一些错误: EF 4: Removing child object from collection does not delete it - why?

当我从父母那里删除一个孩子,当我打电话给SaveChanges()时孩子被删除时,它会给出以下错误信息:

  

操作失败:无法更改关系,因为一个或多个外键属性不可为空。当对关系进行更改时,相关的外键属性将设置为空值。如果外键不支持空值,则必须定义新关系,必须为外键属性分配另一个非空值,或者必须删除不相关的对象。

但是使用DbContext和EF 4.1,“context.DeleteObject(recipe)”不存在。

有什么建议吗?

[编辑]

    public void UpdateWithAttributes(Model model, IEnumerable<Entity> entities)
    {
        var modelOriginal = this.unitOfWork.Model.GetById(model.IModel);

        this.unitOfWork.Context.Entry(modelOriginal).CurrentValues.SetValues(model);
        UpdateEntityAttributeAssociations(modelOriginal, entities);

        this.unitOfWork.Commit();
    }

    public void UpdateEntityAttributeAssociations(Model model, IEnumerable<Entity> current)
    {
        unitOfWork.Context.Entry(model).Collection(m => m.Entities).Load();
ICollection<Entity> original = model.Entities; // perhaps .ToList() necessary

        // delete
        if (original != null)
        {
            List<Entity> toDelete = GetToDelete(original, current);

            foreach (Entity originalEntityToDelete in toDelete)
            {
                unitOfWork.Context.Entity.Remove(originalEntityToDelete);
            }
        }

        // add, update
        if (current != null)
        {
            foreach (Entity currentEntity in current)
            {
                // No need to set the UpdatedWhen. The trigger on the table will handle that.
                if (original.Where(originalEntity => originalEntity.IEntity == currentEntity.IEntity).FirstOrDefault() == null)
                {
                    model.Entities.Add(currentEntity);
                }
            }
        }
    }

2 个答案:

答案 0 :(得分:2)

你必须致电:

context.Recipes.Remove(recipe);

RecipesDbSet<Recipe>的位置。

答案 1 :(得分:1)

我想您再次添加刚刚删除的内容,因为从DbSet中删除也会从子集original中删除实体。然后在第二个循环中再次添加实体。你可以试着像这样抓住这种情况:

public void UpdateEntityAttributeAssociations(Model model,
                                              IEnumerable<Entity> current)
{
    unitOfWork.Context.Entry(model).Collection(m => m.Entities).Load();
    ICollection<Entity> original = model.Entities; // perhaps .ToList() necessary

    // delete
    List<Entity> toDelete = null;
    if (original != null)
    {
        toDelete = GetToDelete(original, current);
        foreach (Entity originalEntityToDelete in toDelete)
        {
            unitOfWork.Context.Entity.Remove(originalEntityToDelete);
        }
    }

    // add, update
    if (current != null)
    {
        foreach (Entity currentEntity in current)
        {
            if (toDelete == null || !toDelete.Contains(currentEntity))
            {
                if (original.Where(originalEntity => originalEntity.IEntity == 
                               currentEntity.IEntity).FirstOrDefault() == null)
                {
                    model.Entity.Add(currentEntity);
                }
            }
        }
    }
}

我也用显式加载子集合替换了你的第一行,因为你手动查询子实体的过程看起来不熟悉,但我不认为这是问题(但我不知道)

修改

我不确定上面的代码是否有助于删除您遇到的异常。我不知道如果你将一个实体添加到已经处于Deleted状态的集合中会发生什么,如果这可能会抛出此异常。

如果代码没有帮助,您可以在没有第二个(插入)循环的情况下测试是否也有问题。如何看待GetToDelete?如何查看您的ModelEntity课程及其关系?并且model.Entity确实是另一个集合而不是model.Entities(或者它是一个错字)?