Что такое TPL-эквивалент rx Observable.FromEventPattern? - PullRequest
0 голосов
/ 24 февраля 2012

В rx вы можете написать:

var oe = Observable.FromEventPattern<SqlNotificationEventArgs>(sqlDep, "OnChange");

и затем подписаться на наблюдаемое, чтобы преобразовать событие OnChange объекта sqlDep в наблюдаемое.

Аналогично, как вы можете создатьЗадача из события C # с использованием библиотеки параллельных задач?

РЕДАКТИРОВАТЬ: уточнение Решение, указанное Дрю и затем явно написанное пользователем 3775487, работает для одного события.Как только задача завершена ... хорошо, она закончена.

Наблюдаемое событие может инициироваться снова в любое время.Это можно рассматривать как наблюдаемый поток.Этакий ISourceBlock в потоке данных TPL.Но в документе http://msdn.microsoft.com/en-us/library/hh228603(v=vs.110).aspx нет примера ISourceBlock.

В конце концов я нашел сообщение на форуме, объясняющее, как это сделать: http://social.msdn.microsoft.com/Forums/en/tpldataflow/thread/a10c4cb6-868e-41c5-b8cf-d122b514db0e

public static ISourceBlock CreateSourceBlock (Action, Action, Action, ISourceBlock> executor) {var bb = newBufferBlock ();executor (t => bb.Post (t), () => bb.Complete (), e => bb.Fault (e), bb);возврат бб;}

//Remark the async delegate which defers the subscription to the hot source.
var sourceBlock = CreateSourceBlock<SomeArgs>(async (post, complete, fault, bb) =>
{
    var eventHandlerToSource = (s,args) => post(args);
    publisher.OnEvent += eventHandlerToSource;
    bb.Complete.ContinueWith(_ => publisher.OnEvent -= eventHandlerToSource);
});

Я не пробовал приведенный выше код.Может быть несоответствие между асинхронным делегатом и определением CreateSourceBlock.

Ответы [ 2 ]

1 голос
/ 24 февраля 2012

Вы можете использовать TaskCompletionSource.

public static class TaskFromEvent
{
    public static Task<TArgs> Create<TArgs>(object obj, string eventName)
        where TArgs : EventArgs
    {
        var completionSource = new TaskCompletionSource<TArgs>();
        EventHandler<TArgs> handler = null;

        handler = new EventHandler<TArgs>((sender, args) =>
        {
            completionSource.SetResult(args);
            obj.GetType().GetEvent(eventName).RemoveEventHandler(obj, handler);
        });

        obj.GetType().GetEvent(eventName).AddEventHandler(obj, handler);
        return completionSource.Task;
    }
}

Пример использования:

public class Publisher
{
    public event EventHandler<EventArgs> Event;

    public void FireEvent()
    {
        if (this.Event != null)
            Event(this, new EventArgs());
    }
}

class Program
{
    static void Main(string[] args)
    {
        Publisher publisher = new Publisher();
        var task = TaskFromEvent.Create<EventArgs>(publisher, "Event").ContinueWith(e => Console.WriteLine("The event has fired."));
        publisher.FireEvent();
        Console.ReadKey();
    }
}

РЕДАКТИРОВАТЬ Ниже приведен пример того, как добитьсяцель с TPL DataFlow.

public class EventSource
{
    public static ISourceBlock<TArgs> Create<TArgs>(object obj, string eventName)
        where TArgs : EventArgs
    {
        BufferBlock<TArgs> buffer = new BufferBlock<TArgs>();
        EventHandler<TArgs> handler = null;

        handler = new EventHandler<TArgs>((sender, args) =>
        {
            buffer.Post(args);
        });

        buffer.Completion.ContinueWith(c =>
            {
                Console.WriteLine("Unsubscribed from event");
                obj.GetType().GetEvent(eventName).RemoveEventHandler(obj, handler);
            });

        obj.GetType().GetEvent(eventName).AddEventHandler(obj, handler);
        return buffer;
    }
}

public class Publisher
{
    public event EventHandler<EventArgs> Event;

    public void FireEvent()
    {
        if (this.Event != null)
            Event(this, new EventArgs());
    }
}

class Program
{
    static void Main(string[] args)
    {
        var publisher = new Publisher();
        var source = EventSource.Create<EventArgs>(publisher, "Event");
        source.LinkTo(new ActionBlock<EventArgs>(e => Console.WriteLine("New event!")));
        Console.WriteLine("Type 'q' to exit");
        char key = (char)0;
        while (true)
        {
            key = Console.ReadKey().KeyChar;             
            Console.WriteLine();
            if (key == 'q') break;
            publisher.FireEvent();
        }

        source.Complete();
        Console.ReadKey();
    }
}
1 голос
/ 24 февраля 2012

Не существует прямого эквивалента для Асинхронного шаблона событий (EAP), запеченного в TPL. Вам нужно использовать TaskCompletionSource<T>, который вы сигнализируете себе в обработчике событий. Посмотрите этот раздел на MSDN , чтобы увидеть пример того, как это будет выглядеть, при котором для демонстрации шаблона используется WebClient :: DownloadStringAsync.

...