我正在尝试为教育目的创建简单的.net编译器。在解析,扫描和构建AST后,我使用Reflection.Emit.ILGenerator
生成.net程序集。
以下是我的程序集生成示例代码:
static void Main(string[] args)
{
string moduleName = "Test.exe";
AssemblyName assemblyName = new AssemblyName(Path.GetFileNameWithoutExtension(moduleName));
AssemblyBuilder assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Save);
ModuleBuilder moduleBuilder = assemblyBuilder.DefineDynamicModule(moduleName);
TypeBuilder typeBuilder = moduleBuilder.DefineType("Program", TypeAttributes.Public);
MethodBuilder mainMethod = typeBuilder.DefineMethod(
"Main", MethodAttributes.Public | MethodAttributes.Static, typeof(void), System.Type.EmptyTypes);
ILGenerator il = mainMethod.GetILGenerator();
il.Emit(OpCodes.Ldstr, "Test!");
il.Emit(OpCodes.Call, typeof(System.Console).GetMethod("WriteLine", new System.Type[] { typeof(string) }));
il.Emit(OpCodes.Ret);
typeBuilder.CreateType();
moduleBuilder.CreateGlobalFunctions();
assemblyBuilder.SetEntryPoint(mainMethod);
assemblyBuilder.Save(moduleName);
}
一切正常,此代码使用以下类生成可执行文件:
using System;
public class Program
{
public Program()
{
}
public static void Main()
{
Console.WriteLine("Test!");
}
}
接下来,我正在使用构建到ThirdPartyLibrary.dll的单个类创建简单的第三方库:
using System;
namespace ThirdPartyLibrary
{
public class MyPrint
{
public static void Print(string s)
{
Console.WriteLine("Third party prints: " + s);
}
}
}
现在我希望将Console.WriteLine
方法调用替换为来自我的库的MyPrint.Print
方法调用,并获得类似的结果代码:
using System;
using ThirdPartyLibrary;
public class Program
{
public Program()
{
}
public static void Main()
{
MyPrint.Print("Test!");
}
}
据我所知,我必须阅读我的ThirdPartyLibrary.dll
文件,然后以某种方式反映它以获取所有类型,然后就可以使用MyPrint
类型。最后,我希望能够将引用路径作为myCompiler.exe参数,例如使用csc.exe。
所以问题是:
答案 0 :(得分:4)
这里唯一的变化是目标方法:
var targetMethod = Assembly.LoadFrom("ThirdPartyLibrary.dll")
.GetType("ThirdPartyLibrary.MyPrint")
.GetMethod("Print", new [] {typeof(string)});
...
il.Emit(OpCodes.Ldstr, "Test!");
il.Emit(OpCodes.Call, targetMethod);
il.Emit(OpCodes.Ret);
因此,我们不是使用Console.WriteLine
,而是调用MyPrint.Print
。
IKVM.Reflection.dll
,因为它允许我进行跨框架定位,但您可能不需要它。您可能找到的便利是Sigil,它通过在您生成时为您提供合理的错误报告来消除IL生成(不平衡堆栈等)的许多痛苦(而不是比你执行时)。 Mono.Cecil也很有趣