Как MSIL вызывать обработчик событий базового класса? - PullRequest
4 голосов
/ 06 ноября 2008

У меня есть класс с именем 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 для вызова обработчика событий базового класса?

Ответы [ 3 ]

6 голосов
/ 06 ноября 2008

Вот 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 голос
/ 06 ноября 2008

Я был действительно очень близок - проблема заключалась в том, что я не загружал аргумент '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);

Теперь все работает нормально.

0 голосов
/ 06 ноября 2008

Использование

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 , и вы должны передать тип возвращаемого значения (в данном случае я считаю нуль) и передать типы аргументов - я думаю, "new Type [] {typeof (EventArgs)}

...