Язык программирования C#9 и платформа .NET5 - Троелсен Эндрю
На заметку! В настоящей главе вопросы применения среды DLR для интеграции с динамическими языками не обсуждаются.
Роль деревьев выражений
Для описания динамического вызова в нейтральных терминах среда DLR использует деревья выражений. Например, взгляните на следующий код С#:
dynamic d = GetSomeData();
d.SuperMethod(12);
В приведенном выше примере среда DLR автоматически построит дерево выражения, которое по существу гласит: "Вызвать метод по имени
SuperMethod()
d
12
Далее запрос отображается на необходимую структуру вызовов для целевого объекта. Деревья выражений обладают одной замечательной характеристикой (помимо того, что их не приходится создавать вручную): они позволяют писать фиксированный оператор кода C# и не беспокоиться о том, какой будет действительная цель.
Динамический поиск в деревьях выражений во время выполнения
Как уже объяснялось, среда DLR будет передавать деревья выражений целевому объекту; тем не менее, на этот процесс отправки влияет несколько факторов. Если динамический тип данных указывает в памяти на объект СОМ, то дерево выражения отправляется реализации низкоуровневого интерфейса СОМ по имени
IDispatch
dynamic
Если динамические данные не указывают на объект СОМ, тогда дерево выражения может быть передано объекту, реализующему интерфейс
IDynamicObject
Наконец, если динамические данные указывают на объект, который не является объектом СОМ и не реализует интерфейс
IDynamicObject
После того как дерево выражения обработано определенным связывателем, динамические данные преобразуются в реальный тип данных в памяти, после чего вызывается корректный метод со всеми необходимыми параметрами. Теперь давайте рассмотрим несколько практических применений DLR, начав с упрощения вызовов .NET Core с поздним связыванием.
Упрощение вызовов с поздним связыванием посредством динамических типов
Одним из случаев, когда имеет смысл использовать ключевое слово
dynamic
Activator.Createlnstance()
System.Reflection
static void CreateUsingLateBinding(Assembly asm)
{
try
{
// Получить метаданные для типа MiniVan.
Type miniVan = asm.GetType("CarLibrary.MiniVan");
// Создать экземпляр MiniVan на лету.
object obj = Activator.CreateInstance(miniVan);
// Получить информацию о TurboBoost.
MethodInfo mi = miniVan.GetMethod("TurboBoost");
// Вызвать метод (null означает отсутствие параметров).
mi.Invoke(obj, null);
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
}
В то время как приведенный код функционирует ожидаемым образом, нельзя не отметить его некоторую громоздкость. Вы должны вручную работать с классом
MethodInfo
dynamic
static void InvokeMethodWithDynamicKeyword(Assembly asm)
{
try
{
// Получить метаданные для типа Minivan.
Type miniVan = asm.GetType("CarLibrary.MiniVan");
<b> // Создать экземпляр MiniVan на лету и вызвать метод.</b>
dynamic obj = Activator.CreateInstance(miniVan);
obj.TurboBoost();
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
}
За счет объявления переменной
obj
dynamic
Использование ключевого слова dynamic для передачи аргументов
Полезность среды DLR становится еще более очевидной, когда нужно выполнять вызовы методов с поздним связыванием, которые принимают параметры. В случае применения "многословных" обращений к рефлексии аргументы нуждаются в упаковке внутрь массива экземпляров
object
Invoke()
MethodInfo