Как создать функцию автоматической привязки для FluentNHibernate на основе атрибутов?

#linq #fluent-nhibernate #nhibernate-mapping

#linq #fluent-nhibernate #nhibernate-отображение

Вопрос:

Я хочу использовать механизм сопоставления, основанный на маркировке атрибута свойства, вот так:

 [AttributeUsage(AttributeTargets.Property | AttributeTargets.Field, AllowMultiple = false, Inherited = true)]
public class DomainSignatureAttribute : Attribute { }

public class SomeThing {
    [DomainSignature]
    public virtual string SomePropertyForMapping {get;set;}
    [DomainSignature]
    public virtual int OtherPropertyForMapping {get;set;}

    public virtual string OtherPropertyWithoutMapping {get;}
}
  

Итак, в дочерних элементах ClassMap я хочу реализовать этот метод, который отображает все свойства, отмеченные атрибутом DomainSignatureAttribute:

 protected void MapPropertiesWithStandartType()
{
    foreach (PropertyInfo property in typeof(T).GetProperties())
    {
        if (property != null)
        {
            object[] domainAttrs = property.GetCustomAttributes(typeof (DomainSignatureAttribute), true);

            if (domainAttrs.Length > 0)
                Map(x => property.GetValue(x, null));
        }
    }
}
  

Но есть небольшая проблема с Linq. Когда FluentNHibernate создает отображение (Cfg.BuildSessionFactory ()), он прерывается с

Пытался добавить свойство ‘GetValue’, когда оно уже добавлено.

исключение. Как я понимаю, мне нужно перестроить выражение Linq: x => property.GetValue(x, null) в правильную форму.

Пожалуйста, не предлагайте использовать функцию автоматической карты и не предлагайте использовать ручное сопоставление.

Комментарии:

1. Вы можете использовать автоматическое сопоставление переопределение автоматического сопоставления (IAutoMappingOverride), а затем определение сопоставления. IgnoreProperty(…) только для свойств, которые вы не хотите сопоставлять.

Ответ №1:

 var domainproperties = typeof(T).GetProperties()
    .Where(prop => prop.GetCustomAttributes(typeof (DomainSignatureAttribute), true).Length > 0);

foreach (var property in domainproperties)
{
    Map(Reveal.Member<T>(property.Name));
}