Получение типа MemberInfo с отражением

Я использую отражение для загрузки древовидной структуры с помощью структуры класса проекта. Каждому из членов класса присваивается специальный атрибут.

У меня нет проблем с получением атрибутов для класса с помощью MemberInfo.GetCustomAttributes(), но мне нужен способ разработки, если член класса является настраиваемым классом, а затем нуждается в синтаксическом анализе для возврата пользовательских атрибутов.

До сих пор мой код:

MemberInfo[] membersInfo = typeof(Project).GetProperties();

foreach (MemberInfo memberInfo in membersInfo)
{
    foreach (object attribute in memberInfo.GetCustomAttributes(true))
    {
        // Get the custom attribute of the class and store on the treeview
        if (attribute is ReportAttribute)
        {
            if (((ReportAttribute)attribute).FriendlyName.Length > 0)
            {
               treeItem.Items.Add(new TreeViewItem() { Header = ((ReportAttribute)attribute).FriendlyName });
            }
        }
        // PROBLEM HERE : I need to work out if the object is a specific type
        //                and then use reflection to get the structure and attributes.
    }
}

Есть ли простой способ получить целевой тип экземпляра MemberInfo, чтобы я мог обработать его соответствующим образом? Я чувствую, что мне не хватает чего-то очевидного, но я собираюсь кругом в минуту.

Ответ 1

GetProperties возвращает массив PropertyInfo, поэтому вы должны использовать его.
Тогда это просто вопрос использования свойства PropertyType.

PropertyInfo[] propertyInfos = typeof(Project).GetProperties();

foreach (PropertyInfo propertyInfo in propertyInfos)
{
    // ...
    if(propertyInfo.PropertyType == typeof(MyCustomClass))
        // ...
}

Ответ 2

Я думаю, что вы можете получить лучшую производительность, если будете носить этот метод расширения:

public static Type GetUnderlyingType(this MemberInfo member)
{
    switch (member.MemberType)
    {
        case MemberTypes.Event:
            return ((EventInfo)member).EventHandlerType;
        case MemberTypes.Field:
            return ((FieldInfo)member).FieldType;
        case MemberTypes.Method:
            return ((MethodInfo)member).ReturnType;
        case MemberTypes.Property:
            return ((PropertyInfo)member).PropertyType;
        default:
            throw new ArgumentException
            (
             "Input MemberInfo must be if type EventInfo, FieldInfo, MethodInfo, or PropertyInfo"
            );
    }
}

Должен работать для любого MemberInfo, а не только PropertyInfo. Вы можете избежать MethodInfo из этого списка, поскольку он не находится под лежащим типом как таковым (но возвращаемым типом).

В вашем случае:

foreach (MemberInfo memberInfo in membersInfo)
{
    foreach (object attribute in memberInfo.GetCustomAttributes(true))
    {
        if (attribute is ReportAttribute)
        {
            if (((ReportAttribute)attribute).FriendlyName.Length > 0)
            {
               treeItem.Items.Add(new TreeViewItem() { Header = ((ReportAttribute)attribute).FriendlyName });
            }
        }

        //if memberInfo.GetUnderlyingType() == specificType ? proceed...
    }
}

Интересно, почему это не было частью BCL по умолчанию.