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