Egg*_*ggi 4 inheritance entity-framework multiple-inheritance
我尝试使用TPC样式的Entity Framework映射一些类,并出现以下错误:
错误:类型"A"无法按定义映射,因为它映射了使用实体拆分或其他形式的继承的类型的继承属性.选择不同的继承映射策略,以便不映射继承的属性,或更改层次结构中的所有类型以映射继承的属性,并且不使用拆分.
使用以下类时发生此错误:
public abstract class BaseEntityTest
public abstract class BaseEntityTest2 : BaseEntityTest
public abstract class BaseEntityTest3 : BaseEntityTest2
public class A: BaseEntityTest3 // this class is the only one with a table in the db
Run Code Online (Sandbox Code Playgroud)
在OnModelCreating方法中,我添加了以下代码以获取TPC映射
modelBuilder.Entity<A>().Map(m =>
{
m.MapInheritedProperties();
m.ToTable("A");
});
Run Code Online (Sandbox Code Playgroud)
当我从结构中排除BaseEntityTest2(这样A只从BaseEntityTest而不是BaseEntityTest2继承)时,错误消失了.这是否意味着无法创建此映射或我只是错过了什么?
编辑:
课程属性:
public abstract class BaseEntityTest
{
[Key]
public Guid Id { get; set; }
public String Info { get; set; }
[Required]
public DateTime CreationDate { get; set; }
[Required]
public String CreationUser { get; set; }
[Required]
public DateTime ModificationDate { get; set; }
[Required]
public String ModificationUser { get; set; }
[ConcurrencyCheck]
[Required]
public int LockVersion { get; internal set; }
}
public abstract class BaseEntityTest2 : BaseEntityTest
{
[Required]
public string Name { get; set; }
public string Description { get; set; }
}
public abstract class BaseEntityTest3: BaseEntityTest2
{
[Required]
public DateTime FromDate { get; set; }
public DateTime ThruDate { get; set; }
}
public class A: BaseEntityTest3{
public String Test { get; set; }
}
Run Code Online (Sandbox Code Playgroud)
EF 4.3.1及更早版本发生错误,但EF 4.4和EF 5.0不发生错误.(EF 4.4实际上是EF 5.0,但是以.NET 4.0作为目标平台.)
但是:只有当您将抽象类用作模型中的实体时才会发生错误,这意味着
你要么DbSet在你的上下文类中使用s,比如
public DbSet<BaseEntityTestX> BaseEntityTestXs { get; set; }
Run Code Online (Sandbox Code Playgroud)或者你有一些Fluent映射BaseEntityTestX,有些modelBuilder.Entity<BaseEntityTestX>()...东西
或者您正在BaseEntityTestX另一个(具体)实体类型中使用其中一个作为导航属性
你需要这个吗?
有一个DbSet<BaseEntityTestX>在你的情况下只会意义,如果你真的想查询的抽象实体,就像之一:
List<BaseEntityTest> list = context.BaseEntityTests
.Where(b => b.Info == "abc").ToList();
Run Code Online (Sandbox Code Playgroud)
结果当然是一个继承自的具体实体的列表BaseEntityTest,但它可以是不同类型的混合,如某些As和某些Bs.你需要这样的询问吗?或者您只想查询一些具体对象:
List<A> list = context.As
.Where(b => b.Info == "abc").ToList();
Run Code Online (Sandbox Code Playgroud)
在后一种情况下,您不需要DbSet抽象基类,也不需要任何继承映射.您可以DbSet<BaseEntityTestX>从上下文类中删除并删除TPC映射,您的错误就会消失.
最后一点 - 具有导航属性到另一个实体中的一个抽象实体 - 对于TPC映射没有意义.它只是不能映射到关系数据库,因为使用TPC映射时,抽象实体没有表,因此外键关系可以从具有navigation属性的具体类的表中引用.
如果将TPC映射扩展到基类,错误也将消失:
modelBuilder.Entity<BaseEntityTestX>().Map(m =>
{
m.MapInheritedProperties();
m.ToTable("BaseEntityTestX");
});
Run Code Online (Sandbox Code Playgroud)
但它会为那些对我来说似乎没有意义的抽象实体创建表格.