Загрузка взаимозависимых сборок из C ++ / CLI
-
21-09-2019 - |
Вопрос
Я хочу загрузить две сборки из C ++ / CLI;сборка A зависит от сборки B, и обе являются VB.Сетевые проекты (3.5).Я хочу, чтобы они загружались из массива байтов, поэтому я использую Assembly::Load() , но когда я пытаюсь создать экземпляр класса из сборки A, фреймворк игнорирует ранее загруженную сборку B и пытается загрузить ее снова, что завершается неудачей, поскольку ее нет в пути поиска."Имя" сборки одно и то же, поэтому я не знаю, почему она выходит из строя.В целях тестирования моя программа загружает байты непосредственно из скомпилированного образа, но реальный код будет загружен по-другому.Это мой тестовый код:
#include "stdafx.h"
using namespace System;
using namespace System::Windows::Forms;
using namespace System::IO;
using namespace System::Reflection;
[STAThreadAttribute]
int main(array<System::String ^> ^args)
{
array<unsigned char>^ bytes;
FileStream^ f;
f = gcnew FileStream(L"c:\\...\\AssemblyB.dll", FileMode::Open);
bytes = gcnew array<unsigned char>((int)f->Length);
f->Read( bytes, 0, (int) f->Length );
f->Close();
f = nullptr;
Assembly^ assemblyb = Assembly::Load(bytes);
f = gcnew FileStream(L"c:\\...\\AssemblyA.dll", FileMode::Open);
bytes = gcnew array<unsigned char>((int)f->Length);
f->Read( bytes, 0, (int) f->Length );
f->Close();
f = nullptr;
Assembly^ assemblya = Assembly::Load(bytes);
bytes = nullptr;
// Here I get the file not found exception!
Object^ mf = assemblya->CreateInstance(L"AssemblyA.MainForm");
// This line is not reached unless I copy assemblyb.dll to my app's folder:
mf->GetType()->InvokeMember(L"ShowDialog",BindingFlags::Default | BindingFlags::InvokeMethod,
mf->GetType()->DefaultBinder, mf, nullptr );
return 0;
}
Ошибка заключается в:
Не удалось загрузить файл или сборку 'AssemblyB, Версия =1.0.3650.39903, Культура = нейтральная, PublicKeyToken=null"или одна из его зависимостей.Система не может найти указанный файл.
Когда я проверяю assemblyb-> Полное имя, там точно написано 'AssemblyB, Версия =1.0.3650.39903, Культура = нейтральная, PublicKeyToken=null'.
Конечно, если я скопирую AssemblyB.dll в папку моей тестовой программы, код будет работать просто отлично, но это не то, что я хочу.
Есть какие-нибудь идеи?
(Кстати, моим вторым шагом будет попытка заставить AssemblyA использовать классы, которые будут доступны в моем C ++ / CLI exe.)
Решение
Ладно, я просто поставил себя в неловкое положение.Все это есть в документах.
// This class is just for holding a managed static variable for assemblyB
ref class Resolver {
public:
static Assembly^ assemblyB;
};
// This is the delegate for resolving assemblies
Assembly^ ResolveHandler(Object^ Sender, ResolveEventArgs^ args)
{
// Warning: this should check the args for the assembly name!
return Resolver::assemblyB;
}
.
.
.
[STAThreadAttribute]
int main(array<System::String ^> ^args)
{
// Set up the handler for the AssemblyResolve event
AppDomain::CurrentDomain->AssemblyResolve += gcnew ResolveEventHandler( ResolveHandler );
.
.
.
// Load assemblyb into the static variable available to the resolver delegate
Resolver::assemblyb = Assembly::Load(bytes);
.
.
.
Я надеюсь, что кто-то найдет это полезным.:)