C#Entity Framework 6 - AutoMapper 6 - 集合/列表 - 添加和更新有效但删除项会引发错误

时间:2017-07-07 12:39:02

标签: c# entity-framework entity-framework-6 automapper automapper-6

正如标题所说我可以添加和更新但是当涉及到删除时我收到错误。

  

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

我知道在这种情况下Description只获取一个空外键但永远不会被删除。我已经看到了一些示例,他们建议循环遍历每个子项并逐个删除它们。 Imao我认为应该有更好的方法。我正在寻找的是一个影响最小的解决方案,只需告诉EF删除整个项目,而不仅仅是取消外键。

https://stackoverflow.com/a/5540956/3850405

使用AutoMapper,AutoMapper.Collection和AutoMapper.Collection.EntityFramework。

控制器方法:

public async Task<IHttpActionResult> UpdateArticle(ArticleViewModel articleVm)
{
    Article articleOriginal = await iArticleRepository.GetAsync(articleVm.Id);
    Article updatedArticle = Mapper.Map<ArticleViewModel, Article>(articleVm, articleOriginal);
    await iArticleRepository.UpdateAsync(updatedArticle);
    return Ok();
}

映射:

Mapper.Initialize(cfg =>
{
    cfg.AddCollectionMappers();

    cfg.SetGeneratePropertyMaps<GenerateEntityFrameworkPrimaryKeyPropertyMaps<DbContext>>();

    cfg.CreateMap<ArticleViewModel, Article>(MemberList.Source)
        .EqualityComparison((src, dst) => src.Id == dst.Id);

    cfg.CreateMap<DescriptionViewModel, Description>(MemberList.Source)
        .EqualityComparison((src, dst) => src.Id == dst.Id);
}
Mapper.AssertConfigurationIsValid();

的ViewModels:

public class ArticleViewModel
{
    public int Id { get; set; }

    public string Name { get; set; }

    public List<DescriptionViewModel> Descriptions { get; set; }
}

public class DescriptionViewModel
{
    public int Id { get; set; }

    public string Heading { get; set; }
}

型号:

public class Article : IEntity<int>
{
    public Article()
    {
        Descriptions = new List<Description>();
    }

    [Key]
    public int Id { get; set; }

    public DateTime Created { get; set; }

    public DateTime Updated { get; set; }

    [MaxLength(256)]
    public string Name { get; set; }

    public virtual ICollection<Description> Descriptions { get; set; }
}

public class Description: IEntity<int>
{
    [Key]
    public int Id { get; set; }

    public DateTime Created { get; set; }

    public DateTime Updated { get; set; }

    [MaxLength(256)]
    public string Heading { get; set; }

    public int ArticleId { get; set; }

    public virtual Article Article { get; set; }
}

1 个答案:

答案 0 :(得分:0)

从这个答案中得到解决方案

https://stackoverflow.com/a/32983252/3850405

和这个博客:

http://www.kianryan.co.uk/2013/03/orphaned-child/

代码:

public class Description: IEntity<int>
{
    [Key, Column(Order = 0), DatabaseGenerated(DatabaseGeneratedOption.Identity)]
    public int Id { get; set; }

    public DateTime Created { get; set; }

    public DateTime Updated { get; set; }

    [MaxLength(256)]
    public string Heading { get; set; }

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

    public virtual Article Article { get; set; }
}

我真的建议您阅读Mosh's answer关于撰写聚合之间的区别,因为它可以帮助您更好地理解EF。