在IlGenerator中调用其他方法

时间:2016-06-15 15:18:58

标签: c# reflection reflection.emit

我通过TypeBuilder构建自己的类型,并尝试添加此方法,调用从不同对象收集的methodInfo

问题是我不知道如何使用ILGenerator.EmitILGenerator.EmitCall

我尝试使用il.EmitCall(OpCodes.Call, methodInfo, arguments)il.Emit(OpCodes.Call, methodInfo),但两者都没有奏效。我总是遇到这个错误:

[InvalidProgramException: Common Language Runtime detected an invalid program.]
   MyImplementationController.Hello1() +0

[TargetInvocationException: Exception has been thrown by the target of an invocation.]
   System.RuntimeMethodHandle.InvokeMethod(Object target, Object[] arguments, Signature sig, Boolean constructor) +0
   System.Reflection.RuntimeMethodInfo.UnsafeInvokeInternal(Object obj, Object[] parameters, Object[] arguments) +192
   System.Reflection.RuntimeMethodInfo.Invoke(Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture) +155
   System.Reflection.MethodBase.Invoke(Object obj, Object[] parameters) +19

这是我的代码:

        foreach (var methodInfo in methodInfosFromSomewhere)
        {
            var arguments = methodInfo.GetParameters().Select(a => a.ParameterType).ToArray();
            MethodBuilder mb = tb.DefineMethod(
                methodInfo.Name,
                MethodAttributes.Final | MethodAttributes.Public,
                CallingConventions.HasThis,
                methodInfo.ReturnType,
                arguments);

            // method 
            ILGenerator il = mb.GetILGenerator();
            int numParams = arguments.Length;
            for (byte x = 0; x < numParams; x++)
            {
                //il.Emit(OpCodes.Ldarg_S, x);
                il.Emit(OpCodes.Ldstr, x);
            }
            //il.EmitCall(OpCodes.Call, methodInfo, arguments);
            il.Emit(OpCodes.Call, methodInfo);

            il.Emit(OpCodes.Ret);
        }

@Edit

最后我知道(可能)问题出在哪里!当我拨打Emit.Call时,我不想在此对象中调用方法。我想从另一个对象调用方法。

请看这个例子:

// this is interface that we want to 'decorate'
public interface IMyInterface
{
    MyResponse Hello1();
    MyResponse Hello2(MyRequest request);
    MyResponse Hello3(MyRequest request);
}
public class MyImplementation : IMyInterface
{
    public MyResponse Hello1()
    {
        return new MyResponse { Name = "empty" };
    }
    // ... rest of implementation, it doesn't matter
}

我想要生成的类是这样的:

public class GeneratedClass : ApiController
{
    public MyInterface myImplementation { get; set; }
    public MyResponse Hello1()
    {
        return myImplementation.Hello1();
    }
    // ... rest of implementation, it doesn't matter
}

如您所见,我想从其他对象调用方法。我知道如何为对象创建属性,但我不知道如何从其他对象调用方法

1 个答案:

答案 0 :(得分:1)

从来源:(http://referencesource.microsoft.com/#mscorlib/system/reflection/emit/ilgenerator.cs,3e110f4a19d1c05e

public virtual void Emit(OpCode opcode, MethodInfo meth)
{
    //...
    if (opcode.Equals(OpCodes.Call) || opcode.Equals(OpCodes.Callvirt) || opcode.Equals(OpCodes.Newobj))
    {
        EmitCall(opcode, meth, null);
    }
    else
    {
        // ...
    }
}

如您所见,如果Emit()EmitCall()OpCodeCall,则Callvirt会调用Newobj,因此{{1} }} Emit()不应该有所作为。

使用EmitCall()发出需要OpCodes.Ldstr类型的操作数。您要做的是在发出string指令之前逐个将参数加载到堆栈。​​

而不是:

OpCodes.Call

试试这个:

for (byte x = 0; x < numParams; x++)
{
    il.Emit(OpCodes.Ldstr, x);
}

问题更新后编辑: 您必须在新类型中定义属性switch (numParams) { case 0: break; case 1: il.Emit(OpCodes.Ldarg_0); break; case 2: il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldarg_1); break; case 3: il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldarg_1); il.Emit(OpCodes.Ldarg_2); break; default: il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Ldarg_1); il.Emit(OpCodes.Ldarg_2); il.Emit(OpCodes.Ldarg_3); for (int i = 4; i < numParams; i++) { il.Emit(OpCodes.Ldarg, mb.GetParameters()[i]); } break; }

试试这个:

myImplementation