NHibernate将实体映射到通用查找表

本文关键字:查找 实体 映射 NHibernate | 更新日期: 2023-09-27 18:01:07

我正在针对一个遗留数据库构建一个应用程序,该数据库将查找值存储在一个通用表中(实际上它包含在四个不同的表中。(这意味着实体表存储查找值的"id",而元数据表则保存该值的"description"。

元数据表如下所示:

  • 表格信息
  • ColumnInfo
  • 商业信息
  • 查找描述信息

要获得查找描述,请连接所有四个表,并指定表名、列名和查找id。查找描述信息表包含两列——一列用于文本值,一列用于数值。

我想为每个查找类型都有一个单独的类(例如,我的Widget类将与基于Widget.WidgetTypeId值的"WidgetType"有多对一关系。(实现这一点的一些策略是什么?该数据模型被1000多个RPG程序使用,因此无法更改。

NHibernate将实体映射到通用查找表

我遇到了与您几乎完全相同的问题,并发现以下解决方案是可行的。

创建SQL视图

-- I'm guessing at the table join structure here
create view LookupView
as
select t.TableName, 
   ci.ColumnName,
   bi.Id, --This ID column needs to be the one used as the FK from other tables
   bi.*, --Or whatever columns you need
   coalesce(di.TextDescription, di.NumericDescription) as Description
from TableInfo t
join ColumnInfo ci on t.Id=ci.TableId
join BusinessInfo bi on bi.Id=ci.BusinessId
join LookupDescriptionInfo di on di.id=ci.id

创建基本查找类

public class Lookup {
    public virtual string Tablename {get; set;}
    public virtual string ColumnName {get; set;}
    public virtual string Description {get; set;}
    public virtual int Id {get; set;}
    //Other BusinessInfo properties
}

创建继承的LookupClass

public class ArmourLookup : Lookup{}

在业务对象上使用ArmourLookup类。

public class HeroArmour{
    //Usual properties etc....
    public virtual ArmourLookup Lookup {get; set;}
}

创建子类区分映射集

public class LookupMap : ClassMap<Lookup> {
    public LookupMap(){
        Id(x=>x.Id).GeneratedBy.Assigned(); //Needs to be a unique ID
        Map(x=>x.Tablename);
        Map(x=>x.ColumnName);
        Map(x=>x.Description);
        //Business Info property mappings here
        Table("LookupView")
        DiscriminateSubClassesOnColumn<string>("ColumnName");
        ReadOnly();
    }
}
public class ArmourLookupMap : SubClassMap<ArmourLookup> {
    public ArmourLookupMap (){
        DiscriminatorValue("ArmourColumn");
    }
}

现在,您可以轻松地为创建新类型的每一列重复子类映射。这里的问题是,您无法在视图中更新或插入新的查找,因此您处于只读模式。

此方法使用列名作为鉴别器,因此不使用表名,但如果查找表中有重复的列名,则可以为每个表创建一个基本查找类,并在映射中指定筛选条件。


另一个潜在的解决方案可以是使用T4模板从查找表生成的枚举。尽管这也是一种只读方法。


您还可以将每个查找表映射为一个类,并使用鉴别器模式从ColumnInfo表中获取不同的类型。

public class TableInfo {
     public virtual int Id {get; set;} 
     public virtual string Tablename {get; set;}
     public IList<ColumnInfo> Columns {get; set;}
}
public class ColumnInfo {
   public virtual int Id {get; set;}
   public virtual TableInfo TableInfo {get; set;}
   public virtual BusinessInfo BusinessInfo {get; set;}
   public virtual LookupDescriptionInfo LookupDescriptionInfo {get; set;}
   //Other properties
}
public class ArmourInfoColumn : ColumnInfo {
    //In the mapping you would discriminate on the columnname column.
}
etc...

同样,如果列信息表中有重复的列名,但表ID不同,则可以选择区分某些XTable类。

您也可以在ColumnType(数字或文本(上进行区分,并将LookupDescription类的子类用于"Description"属性的不同列。

如果你能提供你的表格结构和一些样本值,我可以为你丰富这些想法。