// Применение типа AssemblyName для определения отображаемого имени.
AssemblyName asmName;
asmName = new AssemblyName();
asmName.Name = "CarLibrary";
Version v = new Version("1.0.0.0");
asmName.Version = v;
Assembly a = Assembly.Load(asmName);
Чтобы загрузить сборку .NET Framework (не .NET Core), в параметре
Assembly.Load()
должно быть указано значение
PublicKeyToken
. В .NET Core это не требуется из-за того, что назначение строгих имен используется реже. Например, создайте новый проект консольного приложения по имени
FrameworkAssemblyViewer
, имеющий ссылку на пакет
Microsoft.EntityFrameworkCore
. Как вам уже известно, это можно сделать в интерфейсе командной строки .NET 5 (CLI):
dotnet new console -lang c# -n FrameworkAssemblyViewer
-o .\FrameworkAssemblyViewer -f net5.0
dotnet sln .\Chapter17_AllProjects.sln add .\FrameworkAssemblyViewer
dotnet add .\FrameworkAssemblyViewer
package Microsoft.EntityFrameworkCore -v 5.0.0
Вспомните, что в случае ссылки на другую сборку копия этой сборки помещается в выходной каталог ссылаемого проекта. Скомпилируйте проект с применением CLI:
dotnet build
После создания проекта, добавления ссылки на
EntityFrameworkCode
и компиляции проекта сборку теперь можно загрузить и инспектировать. Поскольку количество типов в данной сборке довольно велико, приложение будет выводить только имена открытых перечислений, используя простой запрос LINQ:
using System;
using System.Linq;
using System.Reflection;
Console.WriteLine("***** The Framework Assembly Reflector App *****\n");
// Загрузить Microsoft.EntityFrameworkCore.dll
var displayName =
"Microsoft.EntityFrameworkCore, Version=5.0.0.0,
Culture=\"\", PublicKeyToken=adb9793829d
dae60";
Assembly asm = Assembly.Load(displayName);
DisplayInfo(asm);
Console.WriteLine("Done!");
Console.ReadLine();
private static void DisplayInfo(Assembly a)
{
Console.WriteLine("***** Info about Assembly *****");
Console.WriteLine($"Asm Name: {a.GetName().Name}" ); // Имя сборки
Console.WriteLine($"Asm Version: {a.GetName().Version}"); // Версия сборки
Console.WriteLine($"Asm Culture:
{a.GetName().CultureInfo.DisplayName}"); // Культура сборки
Console.WriteLine("\nHere are the public enums:");
// Список открытых перечислений.
// Использовать запрос LINQ для нахождения открытых перечислений.
Type[] types = a.GetTypes();
var publicEnums =
from pe in types
where pe.IsEnum && pe.IsPublic
select pe;
foreach (var pe in publicEnums)
{
Console.WriteLine(pe);
}
}
К настоящему моменту вы должны уметь работать с некоторыми основными членами пространства имен
System.Reflection
для получения метаданных во время выполнения. Конечно, необходимость в самостоятельном построении специальных браузеров объектов в повседневной практике вряд ли будет возникать часто. Однако не забывайте, что службы рефлексии являются основой для многих распространенных действий программирования, включая позднее связывание.
Понятие позднего связывания
Позднее связывание представляет собой прием, который позволяет создавать экземпляр заданного типа и обращаться к его членам во время выполнения без необходимости в жестком кодировании факта его существования на этапе компиляции. При построении приложения, в котором производится позднее связывание с типом из внешней сборки, нет причин устанавливать ссылку на эту сборку; следовательно, в манифесте вызывающего кода она прямо не указывается.
На первый взгляд значимость позднего связывания оценить нелегко. Действительно, если есть возможность выполнить "раннее связывание" с объектом (например, добавить ссылку на сборку и выделить память под экземпляр типа с помощью ключевого слова
new
), то именно так следует поступать. Причина в том, что ранее связывание позволяет выявлять ошибки на этапе компиляции, а не во время выполнения. Тем не менее, позднее связывание играет важную роль в любом расширяемом приложении, которое может строиться. Пример построения такого "расширяемого" приложения будет приведен в конце главы, в разделе "Построение расширяемого приложения", а пока займемся исследованием роли класса
Activator
.
Класс System.Activato
Класс
System.Activator
играет ключевую роль в процессе позднего связывания .NET Core. В приведенном далее примере интересен только метод
Activator.Createlnstance()
, который применяется для создания экземпляра типа через позднее связывание. Этот метод имеет несколько перегруженных версий, обеспечивая достаточно высокую гибкость. Самая простая версия метода
CreateInstance()
принимает действительный объект
Туре
, описывающий сущность, которую необходимо разместить в памяти на лету.