基本クラスのイベントハンドラを呼び出すMSILとは何ですか?
-
06-07-2019 - |
質問
次のように、イベントEventConsumedとメソッドOnEventConsumedを定義するEventConsumerというクラスがあります:
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は何ですか?
解決
サンプルアプリの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を実行していないため、インスタンスをロードしていないと思います
他のヒント
私は実際に非常に近かった-問題は、「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);
今では正常に動作します。
使用
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)}
所属していません StackOverflow