Вопрос

Я хочу загрузить две сборки из 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);
.
.
.

Я надеюсь, что кто-то найдет это полезным.:)

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top