我有一个名为EventConsumer的类,它定义了一个EventConsumed事件和一个OnEventConsumed方法,如下所示:
public event EventHandler EventConsumed;
public virtual void OnEventConsumed(object sender, EventArgs e)
{
if (EventConsumed != null)
EventConsumed(this, e);
}
我需要在onEventConsumed运行时添加属性,所以我使用System.Reflection.Emit生成子类。我想要的是MSIL等同于:
public override void OnEventConsumed(object sender, EventArgs e)
{
base.OnEventConsumed(sender, e);
}
到目前为止我所拥有的是:
...
MethodInfo baseMethod = typeof(EventConsumer).GetMethod("OnEventConsumed");
MethodBuilder methodBuilder = typeBuilder.DefineMethod("OnEventConsumed",
baseMethod.Attributes,
baseMethod.CallingConvention,
typeof(void),
new Type[] {typeof(object),
typeof(EventArgs)});
ILGenerator ilGenerator = methodBuilder.GetILGenerator();
// load the first two args onto the stack
ilGenerator.Emit(OpCodes.Ldarg_1);
ilGenerator.Emit(OpCodes.Ldarg_2);
// call the base method
ilGenerator.EmitCall(OpCodes.Callvirt, baseMethod, new Type[0] );
// return
ilGenerator.Emit(OpCodes.Ret);
...
我创建了类型,创建了该类型的实例,并调用它的OnEventConsumed函数,我得到:
Common Language Runtime detected an invalid program.
......这不是很有帮助。我究竟做错了什么?调用基类的事件处理程序的MSIL是什么?
答案 0 :(得分:6)
以下是来自示例应用的IL:
.method public hidebysig virtual instance void OnEventConsumed(object sender, class [mscorlib]System.EventArgs e) cil managed
{
.maxstack 8
L_0000: nop
L_0001: ldarg.0
L_0002: ldarg.1
L_0003: ldarg.2
L_0004: call instance void SubclassSpike.BaseClass::OnEventConsumed(object, class [mscorlib]System.EventArgs)
L_0009: nop
L_000a: ret
}
所以我认为你没有加载实例,因为你没有做ldarg.0
答案 1 :(得分:1)
我实际上非常接近 - 问题是我没有加载'this'参数,并且Callvirt调用子类方法,我实际上想要Call。那部分就变成了:
// load 'this' and the first two args onto the stack
ilGenerator.Emit(OpCodes.Ldarg_0);
ilGenerator.Emit(OpCodes.Ldarg_1);
ilGenerator.Emit(OpCodes.Ldarg_2);
// call the base method
ilGenerator.EmitCall(OpCodes.Call, baseMethod, new Type[0] );
// return
ilGenerator.Emit(OpCodes.Ret);
现在工作正常。
答案 2 :(得分:0)
使用
public virtual void OnEventConsumed(object sender, EventArgs e)
{
if (EventConsumed != null)
EventConsumed(this, e);
}
应该是
public virtual void OnEventConsumed(EventArgs e)
{
EventHandler handler = this.EventConsumed;
if ( null != handler ) handler( this, e );
}
我认为,你应该使用ILGenerator.EmitCalli并且你应该传递一种返回值(在这种情况下我认为是null)并传递参数的类型 - 我认为“new Type [] {typeof(EventArgs) )}