Как передать значение при подписке на событие и получить его при запуске события (проблемы использования DynamicMethod) - PullRequest
0 голосов
/ 10 мая 2010

Задача - создать обработчики событий во время выполнения. Мне нужно, чтобы один метод вызывался с другим значением параметра для разных событий. События и их количество известны только во время выполнения. Поэтому я пытаюсь сгенерировать динамические методы, каждый из которых будет назначен какому-либо событию, но в целом все они просто передают некоторое значение методу экземпляра и вызывают его.

Было бы здорово, если бы нечто подобное можно было сделать более простым способом. Я имею в виду передачу некоторого значения на этапе подписки и последующее получение его при запуске события.

Вот что я пытаюсь сделать сейчас:

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();
    }
}

1 Ответ

0 голосов
/ 12 мая 2010

Лямбды как обработчики событий;)

...