4

イベント 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 は何ですか?

4

3 に答える 3

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を実行していないため、インスタンスをロードしていないと思います。

于 2008-11-06T14:17:23.190 に答える
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);

今では正常に動作します。

于 2008-11-06T14:16:26.827 に答える
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)} と思います

于 2008-11-06T14:11:09.067 に答える