使用Entity Framework TPC进行多重继承

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)

Sla*_*uma 6

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)

但它会为那些对我来说似乎没有意义的抽象实体创建表格.