Вызов метода на поле

Мне просто любопытно, как бы я вызвал метод в поле, используя Emit,

У меня этот класс сгенерирован

public class AClass : IDynamicProxyTestInterface
{
    private DynamicProxy<IDynamicProxyTestInterface> proxy;
    public AClass(DynamicProxy<IDynamicProxyTestInterface> p)
    {
        proxy = p;
    }

    public void Hello()
    {
        // Here I want to do the following
        Int32 code = 123456;
        proxy.HandleCall(code);
    }
}

Закомментированная часть - это то, что я изо всех сил пытался выяснить.

У меня есть этот код (закомментированные вещи - это то, что я включал и выключал, пытаясь заставить его работать)

var methodBuilder = typeBuilder.DefineMethod(
    methodInfo.Name,
    MethodAttributes.Public | MethodAttributes.Virtual,
    methodInfo.ReturnType, 
    methodInfo.GetParameters().Select(p => p.GetType()).ToArray()
    );
var methodIlGen = methodBuilder.GetILGenerator();

// Add proxy call to method
var emptyAction = new DynamicProxyOnBuilder();
ProxyCalls.Add(methodInfo.GetHashCode(), emptyAction);

// Drop method info hash into local variable 0
//var hash = methodInfo.GetHashCode();
//methodIlGen.Emit(OpCodes.Ldc_I4, hash);
//methodIlGen.DeclareLocal(typeof(Int32), false);
//methodIlGen.Emit(OpCodes.Stloc_0);

// Create local variable for instance of this DynamicProxy
methodIlGen.Emit(OpCodes.Ldarg_0);
methodIlGen.Emit(OpCodes.Ldflda, proxyField);
var var2 = methodIlGen.DeclareLocal(this.GetType());
methodIlGen.Emit(OpCodes.Stloc,var2);
////methodIlGen.EmitWriteLine(proxyField);
////methodIlGen.Emit(OpCodes.St);
//methodIlGen.Emit(OpCodes.Ldloc,var2);
methodIlGen.Emit(OpCodes.Call, this.GetType().GetMethods().First(x => x.Name == "HandleCall"));

Есть идеи, что я здесь делаю не так?

Кстати, ошибки, которые я получаю,Common Language Runtime detected an invalid program. а также JIT Compiler encountered an internal limitation.

1 ответ

Решение

Обычно в этих условиях лучше всего написать код на C#, скомпилировать его, а затем проверить скомпилированный IL.

В вашем случае, хотя главное, чего вам не хватает, это то, что ваш метод должен заканчиваться инструкцией "ret", а HandleCall должен быть виртуальным. Так что вам просто нужно (при условии, что HandleCall имеет тип возврата void)

Ldflda [proxyField]   // push the this pointer
Ldc_I4 123            // push the argument
Callvirt [HandleCode] // call the method
Ret                   // return
Другие вопросы по тегам