}
}
Метод
Type.GetCustomAttributes()
возвращает массив объектов со всеми атрибутами, примененными к члену, который представлен объектом
Туре
(булевский параметр управляет тем, должен ли поиск распространяться вверх по цепочке наследования). После получения списка атрибутов осуществляется проход по всем элементам
VehicleDescriptionAttribute
с отображением значения свойства
Description
.
Рефлексия атрибутов с использованием позднего связывания
В предыдущем примере для вывода описания транспортного средства типа Winnebago применялось ранее связывание. Это было возможно благодаря тому, что тип класса
VehicleDescriptionAttribute
определен в сборке
AttributedCarLibrary
как открытый член. Для рефлексии атрибутов также допускается использовать динамическую загрузку и позднее связывание.
Добавьте к решению новый проект консольного приложения по имени
VehicleDescriptionAttributeReaderLateBinding
, установите его в качестве стартового и скопируйте сборку
AttributedCarLibrary.dll
в каталог проекта (или в
\bin\Debug\net5.0
, если вы работаете в Visual Studio). Модифицируйте файл
Program.cs
, как показано ниже:
using System;
using System.Reflection;
Console.WriteLine("***** Value of VehicleDescriptionAttribute *****\n");
ReflectAttributesUsingLateBinding();
Console.ReadLine();
static void ReflectAttributesUsingLateBinding()
{
try
{
<b> // Загрузить локальную копию сборки AttributedCarLibrary.</b>
Assembly asm = Assembly.LoadFrom("AttributedCarLibrary");
<b> // Получить информацию о типе VehicleDescriptionAttribute.</b>
Type vehicleDesc =
asm.GetType("AttributedCarLibrary.VehicleDescriptionAttribute");
<b> // Получить информацию о типе свойства Description.</b>
PropertyInfo propDesc = vehicleDesc.GetProperty("Description");
<b> // Получить все типы в сборке.</b>
Type[] types = asm.GetTypes();
<b> // Пройти по всем типам и получить любые атрибуты VehicleDescriptionAttribute.</b>
foreach (Type t in types)
{
object[] objs = t.GetCustomAttributes(vehicleDesc, false);
<b> // Пройти по каждому VehicleDescriptionAttribute и вывести</b>
<b> // описание, используя позднее связывание.</b>
foreach (object o in objs)
{
Console.WriteLine("-> {0}: {1}\n", t.Name,
propDesc.GetValue(o, null));
}
}
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
}
Если вы прорабатывали примеры, рассмотренные ранее в главе, тогда приведенный код должен быть более или менее понятен. Единственный интересный момент здесь связан с применением метода
PropertyInfo.GetValue()
, который служит для активизации средства доступа к свойству. Вот как выглядит вывод, полученный в результате выполнения текущего примера:
***** Value of VehicleDescriptionAttribute *****
-> Motorcycle: My rocking Harley
-> HorseAndBuggy: The old gray mare, she ain't what she used to be...
-> Winnebago: A very long, slow, but feature-rich auto
Практическое использование рефлексии позднего связывания и специальных атрибутов
Хотя вы видели многочисленные примеры применения этих приемов, вас по-прежнему может интересовать, когда использовать рефлексию, динамическое связывание и специальные атрибуты в своих программах. Действительно, данные темы могут показаться в большей степени относящимися к академической стороне программирования (что в зависимости от вашей точки зрения может быть как отрицательным, так и положительным аспектом). Для содействия в отображении указанных тем на реальные ситуации необходим более серьезный пример. Предположим, что вы работаете в составе команды программистов, которая занимается построением приложения, соблюдая требование о том, что продукт должен быть расширяемым за счет использования добавочных сторонних инструментов.
Что понимается под расширяемостью? Возьмем IDE -среду Visual Studio. Когда это приложение разрабатывалось, в его кодовую базу были вставлены многочисленные "привязки", чтобы позволить другим производителям программного обеспечения подключать специальные модули к IDE - среде. Очевидно, что у разработчиков Visual Studio отсутствовал какой-либо способ установки ссылок на внешние сборки .NET Core, которые на тот момент еще не были созданы (и потому раннее связывание недоступно), тогда как они обеспечили наличие в приложении необходимых привязок? Ниже представлен один из возможных способов решения задачи.
1. Во-первых, расширяемое приложение должно предоставлять некоторый механизм ввода, позволяющий пользователю указать модуль для подключения (наподобие диалогового окна или флага командной строки). Это требует динамической загрузки.
2. Во-вторых, расширяемое приложение должно иметь возможность выяснять, поддерживает ли модуль корректную функциональность (такую как набор обязательных интерфейсов), необходимую для его подключения к среде. Это требует рефлексии.