Я учу себя генерированию промежуточного языка (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