gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-11-30 15:07:28 25 4
gpt4 key购买 nike

我正在针对遗留数据库构建一个应用程序,该数据库将查找值存储在通用表中(实际上它保存在四个不同的表中。)这意味着实体表存储查找值的“id”和元数据表持有这个值的“描述”。

元数据表是这样分解的:

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

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

我想为每种查找类型创建一个单独的类(例如,我的 Widget 类将与基于 Widget.WidgetTypeId 值的“WidgetType”具有多对一的关系。)有哪些策略可以实现这?超过 1000 个 RPG 程序使用该数据模型,因此无法更改。

最佳答案

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

创建 SQL View

-- 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");
}
}

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

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


另一种可能的解决方案是使用 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...

如果您在列信息表中有重复的列名但 tableid 不同,您可以选择区分出一些 XTable 类。

您还可以区分 ColumnType(数字或文本)并将 LookupDescription 类子类化为“描述”属性使用不同的列。

如果您可以提供您的表结构和一些示例值,我可以为您进一步充实这些想法。

关于c# - NHibernate 将实体映射到通用查找表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6549506/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com