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

时间:2012-11-20 20:51:47

标签: c# .net dynamic reflection reflection.emit

这听起来很明显,但我遇到了很多困难。基本上,我正在做的是使用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 个答案:

答案 0 :(得分:4)

如果您正在创建单个方法,那么DynamicMethod是一个更好的选择(特别是因为您的方法是静态的) - 您只需使用CreateDelegate(指定委托类型), cast 到该委托,然后调用。它也减少了开销,并且具有收藏价值。

但是如果你被迫使用MethodBuilder:你必须在CreateType上使用TypeBuilder,然后在现在真实的类型上使用反射(从CreateType)。