Cómo pasar valor cuando la suscripción de evento y obtener cuando se activa el evento (problemas de uso DynamicMethod)

StackOverflow https://stackoverflow.com/questions/2801997

Pregunta

La tarea es crear controladores de eventos en tiempo de ejecución. Necesito un método que se llama con diferente valor de parámetro para diferentes eventos. Los eventos y su número solamente se conocen en tiempo de ejecución. Así que estoy tratando de generar métodos dinámicos, cada uno de los cuales serán asignados a algún evento, pero en general todos ellos sólo tiene que pasar algún valor a un método de instancia y lo llaman.

Sería muy bueno si algo similar podría hacerse de la manera más fácil. Me refiero a pasar algún valor a suscribir etapa y luego obtener cuando se activa el evento.

Esto es lo que estoy tratando de hacer ahora:

public class EventSource
{
    public event EventHandler eventOne;
    public event EventHandler eventTwO;
    public event EventHandler eventThree;
}

public class EventListener
{
    SubscribeForEvents()
    {
        BindingFlags flags =
                    BindingFlags.IgnoreCase |
                    BindingFlags.Public |
                    BindingFlags.Instance;

        // Suppose we've already got EventInfo
        // and target source somewhere
        // so we can do eventInfo.AddEventHandler(target, delegate)
        // Now we need a delegate.

        int value = 42;
        Type tDelegate = eventInfo.EventHandlerType;

        // http://msdn.microsoft.com/en-us/library/ms228976(VS.95).aspx
        Type returnType = GetDelegateReturnType(tDelegate);
        DynamicMethod listener = new DynamicMethod("", null,
            GetDelegateParameterTypes(tDelegate), this.GetType());
        /////////

        Type[] callParameters = { typeof(int) };
        MethodInfo method = this.GetType().GetMethod("ToCallFromDelegate", flags);
        ILGenerator generator = listener.GetILGenerator();

        // No success in this mess. What's wrong?
        generator.Emit(OpCodes.Ldc_I4, value);
        generator.Emit(OpCodes.Call, method);
        generator.Emit(OpCodes.Pop);
        generator.Emit(OpCodes.Ret);
        /////////////

        Delegate delegate = listener.CreateDelegate(tDelegate);
        eventInfo.AddEventHandler(target, delegate);

        // When triggered, there is InvalidProgramException
    }

    void ToCallFromDelegate(int value)
    {
        doSomething();
    }
}
¿Fue útil?

Solución

Lambdas como controladores de eventos;)

Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow
scroll top