Как ссылаться на сборку GAC из динамически генерируемой сборки? - PullRequest
4 голосов
/ 02 июля 2011

Я учу себя генерированию промежуточного языка (IL) в C #, и я застрял на пару часов, которые кажутся похожими днями, как ссылаться на System.Windows.Forms.dll (для пример) из динамической сборки, которую я генерирую, используя AppDomain.CurrentDomain.DefineDynamicAssembly и System.Reflection.Emit ... на основе самого превосходного примера в http://olondono.blogspot.com/2008/02/creating-code-at-runtime.html

У меня работает базовый TransferObjectDllGenerator, но теперь я хотел бы сослаться на существующие библиотеки (только) в сгенерированной сборке и не могу понять, как.

Этот ТАК вопрос привел меня к событию AppDomain.CurrentDomain.AssemblyResolve. Я пытался реализовать обработчик событий, но он никогда не срабатывал, поэтому я думаю, что сделал что-то глупое, например, поместил обработчик событий в совершенно неправильное место?

Будем весьма благодарны за любые указатели в правильном направлении.

Вот моя основная линия ... интересные биты // <<-- Commented thus

using System;
using System.Reflection;
//using System.Windows.Forms; (and removed the project's Reference to System.Windows.Forms)

namespace ILGen
{
/// <summary>
/// Generates .\bin\$whatever\PersonLibrary.dll containing MSIL equivalent to:
///   namespace PersonLibrary {
///     public class Person {
///       public string FirstName { get; set; }
///       public string LastName { get; set; }
///       public Person() { }
///       public Person(string firstname, string lastname) {
///         FirstName = firstname;
///         LastName = lastname;
///       }
///     } //end-class
///   } //end-namespace
/// </summary>
public class Program
{
    public static void Main(String[] args) {
        AppDomain.CurrentDomain.AssemblyResolve += MyAssemblyResolver; // <<-- Hook the "resolve this assembly" event.
        try {
            var dll = new TransferObjectDllGenerator("PersonLibrary");
            dll.AddClass("Person", new[] {
                new Property("FirstName", typeof(string))
              , new Property("LastName", typeof(string))
              , new Property("OK", Type.GetType("System.Windows.Forms.Button")) // <<-- References System.Windows.Forms.dll; Type.GetType returns null.
            });
            Console.WriteLine("Generated: " + dll.Save());
        } finally {
            AppDomain.CurrentDomain.AssemblyResolve -= MyAssemblyResolver; // <<-- Unhook the "resolve this assembly" event.
        }
    }

    static Assembly MyAssemblyResolver(object sender, ResolveEventArgs args) // <<-- Handle the "resolve this assembly" event.
    {
        if ( args.Name.StartsWith("System.Windows.Forms.") ) // <<-- Breakpoint here, which is never reached.
            return Assembly.LoadFrom(@"C:\Windows\winsxs\msil_system.windows.forms_b77a5c561934e089_6.0.6001.22230_none_1a2132e45d2f30fc\System.Windows.Forms.dll");
        return null;
    }
} // end-class
} // end-namespace

Если вам нужен / нужен код TransferObjectDllGenerator, пожалуйста, кричите ... Я не опубликовал его, потому что он слишком большой (IMHO) для сообщения на форуме.

Заранее спасибо за ваше время.

Приветствия. Кит.


РЕДАКТИРОВАТЬ: Чтобы обеспечить рабочий пример для любого, кто найдет эту тему в будущем.

Нет необходимости предоставлять пользовательский обработчик событий AssemblyResolve. Это был phurphy . Нам просто нужно предоставить полное имя сборки ... с именем сборки, пространством имен, версией и GUID.

using System;
using System.Reflection;

namespace ILGen
{
public class Program
{
    public static void Main(String[] args) {
        var dll = new TransferObjectDllGenerator("PersonLibrary"); 
        // We need to provide the fully-qualified-assembly-name to 
        // make the standard assembly-resolver search the GAC. 
        // Thanks to Julien Lebosquain for pointing this out.
        Type buttonType = Type.GetType(
            "System.Windows.Forms.Button, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"); 
        dll.AddClass("Person", new[] {
            new Property("FirstName", typeof(string))
          , new Property("LastName", typeof(string))
          , new Property("OK", buttonType) 
        });
        Console.WriteLine("Generated: " + dll.Save());
    }
} // end-class
} // end-namespace

1 Ответ

5 голосов
/ 02 июля 2011

Type.GetType без указания имени сборки будет искать только имя типа в текущей сборке и в mscorlib.

Попробуйте использовать AssemblyQualifiedName типа. Здесь для .NET 4:

Type.GetType("System.Windows.Forms.Button, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089")

Средство распознавания по умолчанию будет искать в GAC и частном пути к домену приложения, поэтому вам не нужно ничего делать, чтобы получить желаемое поведение. Если вам нужно настроить разрешение, то AssemblyResolve или new Type.GetType overload , использующие функцию распознавателя в .NET 4, - это путь. Обратите внимание, что разрешение вручную из путей GAC или winsxs, как правило, является плохой идеей: пусть платформа выполняет свою работу по разрешению.

...