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

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

у меня нет проблем с получением атрибутов для класса, используя 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, чтобы я мог обрабатывать его соответствующим образом? Я чувствую, что упускаю что-то очевидное, но в данный момент я хожу по кругу.

2 ответов


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

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

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

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

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 по умолчанию.