Using the Task Parallel Library on an event-based asynchronous pattern

后端 未结 2 1532
臣服心动
臣服心动 2020-12-03 08:14

I\'m writing a networked application.

Messages are sent over the transport as such:

Network.SendMessage (new FirstMessage() );

I ca

相关标签:
2条回答
  • 2020-12-03 08:51

    You're right about TaskCompletionSource, it's the key to transforming EAP (event-based asynchronous pattern) to TPL's Task.

    This is documented here: https://docs.microsoft.com/en-us/dotnet/standard/parallel-programming/tpl-and-traditional-async-programming#exposing-complex-eap-operations-as-tasks

    Here is the simplified code:

    public static class Extensions  
    {
        public static Task<XDocument> GetRssDownloadTask(
            this WebClient client, Uri rssFeedUri)
        {
            // task completion source is an object, which has some state.
            // it gives out the task, which completes, when state turns "completed"
            // or else it could be canceled or throw an exception
            var tcs = new TaskCompletionSource<XDocument>(); 
    
            // now we subscribe to completed event. depending on event result
            // we set TaskCompletionSource state completed, canceled, or error
            client.DownloadStringCompleted += (sender, e) => 
            {
                      if(e.Cancelled) 
                      {
                          tcs.SetCanceled();
                      }
                      else if(null != e.Error)
                      {
                          tcs.SetException(e.Error);
                      }
                      else
                      { 
                          tcs.SetResult(XDocument.Parse(e.Result));
                      }
            };
    
            // now we start asyncronous operation
            client.DownloadStringAsync(rssFeedUri);
            // and return the underlying task immediately
            return tcs.Task;
        }
    }
    

    Now, all you need to do, to make a chain of those operations, is just to set your continuations (which is not very comfortable at the moment, and the C# 5 await and async will help alot with it)

    So, this code could be used like this:

    public static void Main()
    {
        var client = new WebClient();
    
        client.GetRssDownloadTask(
            new Uri("http://blogs.msdn.com/b/ericlippert/rss.aspx"))
            .ContinueWith( t => {
                ShowXmlInMyUI(t.Result); // show first result somewhere
                // start a new task here if you want a chain sequence
            });
    
        // or start it here if you want to get some rss feeds simultaneously
    
        // if we had await now, we would add 
        // async keyword to Main method defenition and then
    
        XDocument feedEric = await client.GetRssDownloadTask(
            new Uri("http://blogs.msdn.com/b/ericlippert/rss.aspx"));
        XDocument feedJon = await client.GetRssDownloadTask(
            new Uri("http://feeds.feedburner.com/JonSkeetCodingBlog?format=xml"));
        // it's chaining - one task starts executing after 
        // another, but it is still asynchronous
    }
    
    0 讨论(0)
  • 2020-12-03 08:52

    Jeremy Likness has a blog entry title Coroutines for Asynchronous Sequential Workflows using Reactive Extensions (Rx) that might interest you. Here is the question he tries to answer:

    The concept is straightforward: there are often times we want an asynchronous set of operations to perform sequentially. Perhaps you must load a list from a service, then load the selected item, then trigger an animation. This can be done either by chaining the completed events or nesting lambda expressions, but is there a cleaner way?

    0 讨论(0)
提交回复
热议问题