使用反射获取MemberInfo的类型

29

我正在使用反射来加载一个树形视图,该视图显示项目的类结构。每个类中的成员都有一个自定义属性。

我可以使用 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个回答

74

我认为如果您携带这个扩展方法,您可以获得更好的性能:

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 的默认部分。


1
我喜欢这个!即使我知道它是一个“PropertyInfo”,但总感觉有点靠不住。 - Daniel Park
当 memberInfo 来自 MemberExpression.Member 时,我会得到一个异常,它是一个属性,但无法转换。 - Daniel
@Daniel 看起来中间有一个 UnaryExpression。你需要对 UnaryExpression 进行类型检查,如果是的话就取它的 Operand 属性。最好将其作为一个单独的问题。 - nawfal

13

GetProperties 返回一个 PropertyInfo 数组,因此您应该使用它。
然后只需使用PropertyType 属性即可。

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

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

5
是的,MemberInfo[] membersInfo = 这个代码行有点不好。我喜欢使用 var 的原因之一是减少出错的可能性。 - Marc Gravell

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接