Как пройти значение при подписке на событие и получить его, когда событие срабатывает (проблемы использования DynamicMethod)

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

Вопрос

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

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

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

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();
    }
}
Это было полезно?

Решение

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

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top