Характеристики типа как целого
Класс Type содержит больше трёх десятков методов, позволяющих получать информацию об описываемом типе. Так можно узнать какое-либо свойство типа или наличие (и значение) определённого атрибута, выяснить принадлежность типа к некоторому семейству.
Методы, позволяющие получать информацию о типе, приведены в таблице 2.1.
Вот, например, как можно с помощью этих свойств получить описание семейства типа:
string GetTypeDescription(Type type)
{ return type.IsClass ? "class"
: type.IsInterface ? "interface"
: type.IsArray ? "array"
: type.IsEnum ? "enum"
: type.IsValueType ? "struct"
: "";
}
Таблица 2.1 - Методы, позволяющие получать информацию о типе
IsAbstract | Является ли тип абстрактным классом или интерфейсом |
IsClass | Класс ли это? (т.е. не value Type и не интерфейс) |
IsSealed | Может ли тип иметь наследников? |
IsInterface | Это интерфейс? (Заметьте, интерфейс - это не класс!) |
IsCOMObject | А уж не COM-объект ли скрывается под видом класса .NET? |
IsValueType | Это value type (тип-значение)? |
IsEnum | Это перечисление? |
IsPrimitive | Не является ли тип примитивным (т.е. одним из bool, byte, sbyte, short, ushort, int, uint, long, ulong, char, double, float)? |
IsArray | Является ли тип массивом. Тип элементов массива можно получить с помощью свойства ElementType. |
IsPointer | Является ли тип ссылкой? Тип указываемого объекта можно получить с помощью свойства ElementType. |
IsByRef | А уж не тип ли это параметра, передаваемого по ссылке? Если да, то тип передаваемого объекта можно получить с помощью свойства ElementType. |
HasElementType | HasElementType = IsArray || IsPointer || IsByRef. Т.е. тип является не самостоятельным типом, а лишь "производным от" ElementType. |
Члены класса
Для получения информации о членах класса имеется несколько групп (семейств) методов. Каждая группа обеспечивает получение информации об определённом типе членов класса. Например, информацию только о конструкторах класса, или только о его полях. Но есть одна группа методов, которые позволяют получать информацию сразу обо всех членах класса. Это семейство методов GetMembers.
public MemberInfo[] GetMembers();
public abstract MemberInfo[] GetMembers(BindingFlags);
В каждом семействе имеется не меньше двух методов, аналогичных приведенным выше. Один из этих методов возвращает массив, содержащий информацию об открытых (public) членах, а другой позволяет указать, информацию о каких членах класса следует возвращать. Например, если передать в качестве параметра BindingFlags.Public | BindingFlags.Instance, будут возвращены все открытые нестатические члены класса. Если вам нужна информация ещё и о статических членах, нужно добавить ещё один флаг BindingFlags.Static. А если вас интересуют не только открытые, но и закрытые члены, следует использовать флаг BindingFlags.NonPublic.
Ниже перечислены основные семейства методов класса Type, используемые для получения информации о членах класса.
Таблица 2.2 – Основные методы класса Type
GetConstructors | Получение информации о конструкторах класса. Возвращает массив элементов типа ConstructorInfo. |
GetCustomAttributes | Получение информации об атрибутах класса. Возвращает массив объектов-атрибутов. |
GetEvents | Получение информации о событиях, поддерживаемых классом. Возвращает массив элементов типа EventInfo. |
GetFields | Получение информации о полях класса. Возвращает массив элементов типа FieldInfo. |
GetInterfaces | Получение информации об интерфейсах, реализованных или унаследованных классом. Возвращает массив элементов типа Type. |
GetMembers | Получение информации о членах класса (включая свойства, методы, поля, события и т.д.) Возвращает массив элементов типа MemberInfo. |
GetMethods | Получение информации о методах класса. Возвращает массив элементов типа MethodInfo. |
GetNestedTypes | Получение информации о вложенных типах класса. Возвращает массив элементов типа Type. |
GetProperties | Получение информации о свойствах (properties) класса. Возвращает массив элементов типа PropertyInfo. |
GetTypeArray | Получение информации о типе элементов массива (если рассматриваемый тип является массивом). Возвращает соответствующий объект Type. |
Поскольку и конструкторы, и поля, и события, и методы, и свойства - это всё члены класса, то совершенно логично, что соответствующие им типы метаданных, (ConstructorInfo, EventInfo, FieldInfo, MethodInfo, PropertyInfo) унаследованы от типа MemberInfo.
Давайте рассмотрим пример получения информации обо всех конструкторах класса.
public static string TraceCtorsOf(Type type)
{ string trace = "";
ConstructorInfo[] arrCI = type.GetConstructors();
foreach (ConstructorInfo ci in arrCI)
{ trace += (ci.IsStatic ? "static " : "")
+ (ci.IsPrivate ? "private " : "")
+ (ci.IsFamily ? "protected " : "")
+ (ci.IsAssembly ? "internal " : "")
+ ci.Name;
ParameterInfo[] arrParamInfo = ci.GetParameters();
trace += "(";
for (int i = 0; i != arrParamInfo.Length; i++)
{ ParameterInfo parInf = arrParamInfo[i];
trace += (i != 0 ? ", " : "")
+ (parInf.IsIn ? "in " : "")
+ (parInf.IsOut ? "out " : "")
+ (parInf.IsOptional ? "optional ": "")
+ (parInf.IsLcid ? "lcid " : "")
+ parInf.ParameterType.Name + " "
+ parInf.Name
+ ((parInf.DefaultValue != DBNull.Value)
? (" = " + parInf.DefaultValue) : "")
;
}
trace += ");\r\n";
}
return trace;
}
В этом примере можно видеть, каким способом извлекается информация о параметрах методов. Каждая структура данных, отвечающая за методы-члены (EventInfo, MethodInfo, ConstructorInfo), имеет метод GetParameters(), возвращающий массив элементов типа ParameterInfo. Каждый элемент этого массива описывает отдельный параметр метода: его тип, имя, значение по умолчанию, направление передачи данных (in/out) и др.