我已经开始在C#中测试Fluent NHibernate 我有一个很好的规范化对象结构,有20个相关的类。 我目前使用Fluent 1.3和NHibernate 3.2。 到目前为止,我已经设法使用了适合我的AutoMap功能, 很方便!
但...... 其中3个表是“枚举表”,需要将其记录设置为具有特定的Id值。 我尝试对这些表进行手动映射,然后将其余的表自动化。 但是当创建手动表时,它会失败,因为它引用了一个自动执行的表(并且不适用于手动映射器?)
是否可以使用AutoMapping但是对于一些非常少的类来覆盖主键上的身份创建? 我试图制定一个自定义约定,但没有成功。
public class OverrideIdentityGeneration : Attribute
{
}
public class ConventionIdentity : AttributePropertyConvention<OverrideIdentityGeneration>
{
protected override void Apply(OverrideIdentityGeneration attribute, IPropertyInstance instance)
{
instance.Generated.Never();
}
}
还有其他方法吗? 这将是悲伤被强制回使用手动映射的所有类....
答案 0 :(得分:4)
class MyIdConvention : IIdConvention
{
public void Apply(IIdentityInstance instance)
{
if (instance.EntityType == ...)
{
instance.GeneratedBy.Assigned();
}
}
}
<强>更新强>
对于类似枚举的类,通常更容易将枚举定义为id
class ConfigValue
{
public virtual Config Id { get; set; }
}
// the convention is easy
if (instance.EntityType.IsEnum)
{
instance.GeneratedBy.Assigned();
// to save as int and not string
instance.CustomType(typeof(Config));
}
// querying without magic int values
var configValue = Session.Get<ConfigValue>(Config.UIColor);
答案 1 :(得分:-2)
我使用了Fifo给出的想法并将其扩展为使用自定义属性。 为了使代码可读并在其他约定中使用类似的想法时避免冗余,我添加了一个扩展方法来检查自定义属性。
这是我最终得到的代码:
/// <summary>
/// Convention to instruct FluentNHIbernate to NOT generate identity columns
/// when custom attribute is set.
/// </summary>
public class ConventionIdentity : IIdConvention
{
public void Apply(IIdentityInstance instance)
{
if(instance.CustomAttributeIsSet<NoIdentity>())
instance.GeneratedBy.Assigned();
}
}
/// <summary>
/// Custom attribute definition.
/// </summary>
public class NoIdentity : Attribute
{
}
/// <summary>
/// Example on how to set attribute.
/// </summary>
public class Category
{
[NoIdentity]
public int Id { get; set; }
public string Name { get; set; }
}
public static class IInspectorExtender
{
/// <summary>
/// Extender to make convention usage easier.
/// </summary>
public static T GetCustomAttribute<T>(this IInspector instance)
{
var memberInfos = instance.EntityType.GetMember(instance.StringIdentifierForModel);
if(memberInfos.Length > 0)
{
var customAttributes = memberInfos[0].GetCustomAttributes(false);
return customAttributes.OfType<T>().FirstOrDefault();
}
return default(T);
}
}