Вызов метода на поле
Мне просто любопытно, как бы я вызвал метод в поле, используя 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