我有一项使用 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 方法时,它将永远撤回所有项目:
我的列表方法是这样的:
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 列表,而不是此后的其余场图。
有人可以帮助我解决这个问题吗?
答案 0 :(得分:1)
我的猜测是,由于EF导航属性修复,您一次又一次看到相同的实例。
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": []
}
}
]