不需要时的EF Core加载关系

时间:2019-10-10 12:37:26

标签: entity-framework entity-framework-core eager-loading

我有一项使用 EagerLoading 撤回实体的服务。 在很大程度上,这是可行的,但我最近发现了一个问题。 我有一个字段:

public class Field
{
    [Required] public int Id { get; set; }
    [Required, MaxLength(100)] public string CategoryId { get; set; }
    [Required, MaxLength(100)] public string Name { get; set; }
    public FieldDataType DataType { get; set; }
    public bool IsSpecification { get; set; }

    public IList<FieldMap> FieldMaps { get; set; }
}

如您所见,它具有 FieldMaps 的集合。 FieldMap 看起来像这样:

public class FieldMap
{
    public int Id { get; set; }
    public int FeedId { get; set; }
    public int FieldId { get; set; }
    [Required, MaxLength(100)] public string Path { get; set; }

    public Field Field { get; set; }
}

在我的DbContext中这样映射的是:

modelBuilder.Entity<Field>().HasMany(m => m.FieldMaps).WithOne().HasForeignKey(m => m.FieldId);

// Unique constraints
modelBuilder.Entity<Field>().HasIndex(m => new { m.CategoryId, m.Name, m.IsSpecification }).IsUnique();
modelBuilder.Entity<FieldMap>().HasIndex(m => new { m.FeedId, m.Path }).IsUnique();

// Disable cascade delete
modelBuilder.Entity<FieldMap>().HasOne(m => m.Field).WithMany(m => m.FieldMaps).OnDelete(DeleteBehavior.Restrict);

问题是,当我调用 List 方法时,它将永远撤回所有项目:

enter image description here

我的列表方法是这样的:

public IQueryable<T> List(params string[] includes)
{
    IQueryable<T> query = _dbEntitySet;

    return includes == null ? 
        query : 
        includes.Aggregate(query, (current, include) => current.Include(include));
}

我这样调用它:

[HttpGet("{id:int}/fields")]
[ProducesResponseType(typeof(List<Field>), StatusCodes.Status200OK)]
public IActionResult ListFieldMaps(int id)
{
    var feeds = _feedService.List();
    var feed = feeds.SingleOrDefault(m => m.Id.Equals(id));
    if (feed == null) return NotFound();

    var fieldMaps = _fieldMapService.List("Field");

    if (fieldMaps.Any(m => m.Field == null)) return BadRequest(Resources.NullFieldError);

    return Ok(fieldMaps.Where(m =>
        m.FeedId.Equals(id) &&
        m.Field.IsSpecification == (feed.Type == FeedType.Specification)).ToList());
}

如您所见,它只应加载一个具有一个 Field FieldMaps 列表,而不是此后的其余场图。

有人可以帮助我解决这个问题吗?

2 个答案:

答案 0 :(得分:1)

我的猜测是,由于EF导航属性修复,您一次又一次看到相同的实例。

From the documentation

  

Entity Framework Core将自动修复导航属性   到先前已加载到上下文中的任何其他实体   实例。因此,即使您没有明确包括   导航属性,如果某些或   所有相关实体都是先前加载的。

尝试在Visual Studio中设置对象ID(将变量添加到Watch,在其上单击鼠标右键,然后选择“制作对象ID”)。这将为每个实例创建一个唯一的ID,并告诉您是否要加载不同的实体。

答案 1 :(得分:1)

我找到了这篇文章:

https://docs.microsoft.com/en-us/ef/core/querying/related-data#lazy-loading

和这个问题:

https://github.com/aspnet/EntityFrameworkCore/issues/11564

并且他们两个都可以添加此内容:

services.AddMvc()
    .AddJsonOptions(options =>
        options.SerializerSettings.ReferenceLoopHandling = Newtonsoft.Json.ReferenceLoopHandling.Ignore);

这导致我的问题得以解决。 现在,当我发出请求时,我会收到以下响应:

[
    {
        "Id": 1,
        "FeedId": 1,
        "FieldId": 10,
        "Path": "aw_product_id",
        "Field": {
            "Id": 10,
            "CategoryId": "cameras",
            "Name": "gtin",
            "DataType": 0,
            "IsSpecification": false,
            "FieldMaps": []
        }
    }
]