使用Reflection.Emit创建一个方法,然后调用它

3
这听起来似乎是一件很明显的事情,但我遇到了很多困难。基本上,我正在使用Reflection.Emit生成一个方法,然后想调用它。目前为止,我已经构建了该方法,但是在构建完成后无法获取对该方法的引用,因为“在创建类型之前不支持调用所调用的成员”。
以下是我的基本操作步骤:
AssemblyBuilder assembly;
ModuleBuilder module;
TypeBuilder containerTypeBuilder;
Type containerType;
var name = new AssemblyName();
name.Name = "DynamicWrapper";
var domain = Thread.GetDomain();
assembly = domain.DefineDynamicAssembly(name, AssemblyBuilderAccess.RunAndSave);
module = assembly.DefineDynamicModule(assembly.GetName().Name, false);
containerTypeBuilder = module.DefineType("__DynamicWrapperType",
                              TypeAttributes.Public | TypeAttributes.Class |
                              TypeAttributes.AutoClass |
                              TypeAttributes.AnsiClass | TypeAttributes.BeforeFieldInit |
                              TypeAttributes.AutoLayout, typeof(object));
//build method
var mb = containerTypeBuilder.DefineMethod("generatedmethod" + (unique++),
                              MethodAttributes.Public | MethodAttributes.Static, typeof (int),
                              new Type[] {});
//build method body and all that
.....
var type=module.GetType("__DynamicWrapperType");
var info=type.GetMethod(mb.Name, BindingFlags.Static | BindingFlags.Public); //error here

我该如何将刚刚构建好的方法加载起来以便调用它?

我也尝试过mb.Invoke,但是这会导致"在动态模块中不支持被调用的成员"。

1个回答

4
如果你正在创建单个方法,那么DynamicMethod是更好的选择(特别是当你的方法是静态的)- 你只需要使用CreateDelegate(指定委托类型),转换为该委托,并调用。它也更少开销,且可回收。
但如果你被迫使用MethodBuilder:你必须在TypeBuilder上使用CreateType,然后使用反射在现在的真实类型(从CreateType返回)上操作。

不确定为什么一开始我没有考虑 DynamicMethod。我会看看它是否可用(这只是单元测试,所以不是非常重要的可收集,但听起来更容易使用)。 - Earlz
@Earlz,这样做会简单得多:没有AssemblyBuilder、ModuleBuilder、AssemblyName、TypeBuilder、MethodBuilder,基本上只需要使用ILGenerator。 - Marc Gravell
是的,只用了大约2秒钟就转换为使用DynamicMethod,并解决了所有问题 :) 现在只需要找出是什么导致我的IL抛出无效程序异常。 - Earlz
@Earlz 通常是指不平衡的堆栈,例如一个返回你不需要的值的 Add 操作(在这种情况下,添加一个“pop”操作)。如果你想让我看一下,请告诉我...我已经做了很多 ILGenerator。 - Marc Gravell

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接