我们通过每个层次结构的表存储联系人,联系人要么是一家公司,要么是一个人,一个人总是属于一家公司。两者都继承自联系人。
使用EF Core 2.1。
看起来像这个
public abstract class Contact {
public virtual Guid Id { get; set; }
public virtual ICollection<Source> Sources{ set; get; } = new Collection<Source>();
}
public class Company : Contact {
public string CompanyName { set; get; }
public virtual ICollection<Person> People { set; get; } = new Collection<Person>();
}
public class Person: Contact {
public string Name { set; get; }
public DateTime Birthday { set; get; }
public virtual Company Company { set; get; }
}
到目前为止,我们现在想做的是查询Sources
并包括contacts
(所有这些,无论是个人还是公司(
context.Contact.Include(c => c.Contact).FirstOrDefault();
这会生成以下异常
Unable to cast object of type 'System.DateTime' to type 'System.Nullable``1[System.Guid]'.'
StackTrace
at lambda_method(Closure , ValueBuffer )
at Microsoft.EntityFrameworkCore.ChangeTracking.Internal.InternalMixedEntityEntry..ctor(IStateManager stateManager, IEntityType entityType, Object entity, ValueBuffer& valueBuffer)
at Microsoft.EntityFrameworkCore.ChangeTracking.Internal.InternalEntityEntryFactory.NewInternalEntityEntry(IStateManager stateManager, IEntityType entityType, Object entity, ValueBuffer& valueBuffer)
at Microsoft.EntityFrameworkCore.ChangeTracking.Internal.StateManager.StartTrackingFromQuery(IEntityType baseEntityType, Object entity, ValueBuffer& valueBuffer, ISet`1 handledForeignKeys)
at Microsoft.EntityFrameworkCore.Query.Internal.QueryBuffer.StartTracking(Object entity, IEntityType entityType)
at lambda_method(Closure , QueryContext , AdSourceCrm , Object[] )
at Microsoft.EntityFrameworkCore.Query.Internal.IncludeCompiler._Include[TEntity](QueryContext queryContext, TEntity entity, Object[] included, Action`3 fixup)
at lambda_method(Closure , TransparentIdentifier`2 )
at System.Linq.Enumerable.SelectEnumerableIterator`2.MoveNext()
at System.Linq.Enumerable.TryGetFirst[TSource](IEnumerable`1 source, Boolean& found)
at lambda_method(Closure )
at Microsoft.EntityFrameworkCore.Query.Internal.LinqOperatorProvider.ResultEnumerable`1.GetEnumerator()
at Microsoft.EntityFrameworkCore.Query.Internal.LinqOperatorProvider._TrackEntities[TOut,TIn](IEnumerable`1 results, QueryContext queryContext, IList`1 entityTrackingInfos, IList`1 entityAccessors)+MoveNext()
at Microsoft.EntityFrameworkCore.Query.Internal.LinqOperatorProvider.ExceptionInterceptor`1.EnumeratorExceptionInterceptor.MoveNext()
at System.Linq.Enumerable.TryGetFirst[TSource](IEnumerable`1 source, Boolean& found)
at System.Linq.Enumerable.First[TSource](IEnumerable`1 source)
at Microsoft.EntityFrameworkCore.Query.Internal.QueryCompiler.<>c__DisplayClass15_1`1.<CompileQueryCore>b__0(QueryContext qc)
at System.Linq.Queryable.FirstOrDefault[TSource](IQueryable`1 source)
at EfTest.Program.Main(String[] args) in Program.cs:line 18
我们已经花了几个小时(更像是几天(试图找到原因和解决方案,但没有成功。
EF Core不知何故与我们的构造发生了冲突(也许构造本身是错误的(
如果禁用LazyLoading,问题就会消失。
你知道原因是什么吗?什么能解决这个问题?
编辑2我删除了Sources
,因为它们似乎与问题无关
编辑3我添加了一个样本回购
只需将DbContextFactory中的ConnectionString更改为本地路径即可。
我能够在EF Core 2.1.4和2.2预览中使用提供的repo复制它。
正如您在上次更新中提到的,这个问题在某种程度上与延迟加载(代理?(有关,因为没有UseLazyLoadingProxies()
,代码可以按预期工作(这就是为什么我最初无法复制它(。
由于这显然是EF Core的错误,您只能向EF Core Issue Tracker报告并等待修复。不幸的是,在即将发布的2.2版本中,你可能不会得到它,但谁知道呢,它值得一试。
在我看来,这就像是实体框架中的一个错误,但是,您可以通过显式指定自动生成的属性来防止这种情况发生,例如,将CompanyId属性显式添加到Person类
这将防止EF Core 错误地将Birthday列映射到CompanyId列
public class Person : Base
{
// ...
public virtual Company Company { set; get; }
public Guid? CompanyId { get; set; }
}