3

我正在使用我的可移植类库异步检索一些 rss 文章,该类库使用 Microsoft.Bcl 库(它没有 Task.WhenAll)。每篇文章都有一个我需要异步检索的 rss 评论的 url。

下面的代码是我的库。我调用 GetArticles() 但它不返回任何创建调用 GetComments() 以异步获取评论的任务列表。

我尝试在 GetArticles 中使用 Task.WaitAll 来等待评论,但它不会阻塞线程。任何帮助,将不胜感激。

    private const string ArticlesUri = "";

    public async Task<List<ArticleBrief>> GetArticles()
    {
        var results = new List<ArticleBrief>();
        try
        {
            var wfw = XNamespace.Get("http://wellformedweb.org/CommentAPI/");
            var media = XNamespace.Get("http://search.yahoo.com/mrss/");
            var dc = XNamespace.Get("http://purl.org/dc/elements/1.1/");

            var t = await WebHttpRequestAsync(ArticlesUri);
            StringReader stringReader = new StringReader(t);
            using (var xmlReader = System.Xml.XmlReader.Create(stringReader))
            {
                var doc = System.Xml.Linq.XDocument.Load(xmlReader);
                results = (from e in doc.Element("rss").Element("channel").Elements("item")
                           select
                               new ArticleBrief()
                               {
                                   Title = e.Element("title").Value,
                                   Description = e.Element("description").Value,
                                   Published = Convert.ToDateTime(e.Element("pubDate").Value),
                                   Url = e.Element("link").Value,
                                   CommentUri = e.Element(wfw + "commentRss").Value,
                                   ThumbnailUri = e.Element(media + "thumbnail").FirstAttribute.Value,
                                   Categories = GetCategoryElements(e.Elements("category")),
                                   Creator = e.Element(dc + "creator").Value
                               }).ToList();

            }

            var tasks = new Queue<Task>();
            foreach (var result in results)
            {
                tasks.Enqueue(
                    Task.Factory.StartNew(async ()=>
                        {
                            result.Comments = await GetComments(result.CommentUri);
                        }
                    ));
            }

            Task.WaitAll(tasks.ToArray());
        }
        catch (Exception ex)
        {
            // should do some other
            // logging here. for now pass off
            // exception to callback on UI
            throw ex;
        }

        return results;
    }

    public async Task<List<Comment>> GetComments(string uri)
    {
        var results = new List<Comment>();
        try
        {
            var wfw = XNamespace.Get("http://wellformedweb.org/CommentAPI/");
            var media = XNamespace.Get("http://search.yahoo.com/mrss/");
            var dc = XNamespace.Get("http://purl.org/dc/elements/1.1/");

            var t = await WebHttpRequestAsync(uri);
            StringReader stringReader = new StringReader(t);
            using (var xmlReader = System.Xml.XmlReader.Create(stringReader))
            {
                var doc = System.Xml.Linq.XDocument.Load(xmlReader);
                results = (from e in doc.Element("rss").Element("channel").Elements("item")
                           select
                               new Comment()
                               {
                                   Description = e.Element("description").Value,
                                   Published = Convert.ToDateTime(e.Element("pubDate").Value),
                                   Url = e.Element("link").Value,
                                   Creator = e.Element(dc + "creator").Value
                               }).ToList();
            }
        }
        catch (Exception ex)
        {
            // should do some other
            // logging here. for now pass off
            // exception to callback on UI
            throw ex;
        }

        return results;
    }

    private static async Task<string> WebHttpRequestAsync(string url)
    {
        //TODO: look into getting 
        var request = WebRequest.Create(url);
        request.Method = "GET";

        var response = await request.GetResponseAsync();
        return ReadStreamFromResponse(response);
    }

    private static string ReadStreamFromResponse(WebResponse response)
    {
        using (Stream responseStream = response.GetResponseStream())
        using (StreamReader sr = new StreamReader(responseStream))
        {
            string strContent = sr.ReadToEnd();
            return strContent;
        }
    }

    private List<string> GetCategoryElements(IEnumerable<XElement> categories)
    {
        var listOfCategories = new List<string>();

        foreach (var category in categories)
        {
            listOfCategories.Add(category.Value);
        }

        return listOfCategories;
    }

更新了解决方案的代码,刚刚在 Enqueue 方法上添加了 .UnWrap() :

            var tasks = new Queue<Task>();
            foreach (var result in results)
            {
                tasks.Enqueue(
                    Task.Factory.StartNew(async ()=>
                        {
                            result.Comments = await GetComments(result.CommentUri);
                        }
                    ).Unwrap());
            }

            Task.WaitAll(tasks.ToArray());
4

2 回答 2

4

它正在适当地等待。问题是你正在创建一个Task创建另一个任务的 a (即StartNew返回 aTask<Task>并且你只在等待Task很快完成的外部任务(它在内部任务完成之前完成))。

问题将是:

  • 你真的想要那个内心的任务吗?
    • 如果是,那么您可以使用Task.Unwrap来获取代表内部和外部都完成的代理任务,Task并使用它来等待。
    • 如果不是,那么您可以在 StartNew 中删除async/的使用,await以便没有内部任务(我认为这是首选,不清楚为什么需要内部任务)。
  • 你真的需要对Wait异步任务进行同步吗?阅读 Stephen Cleary 的一些博客:http: //blog.stephencleary.com/2012/02/async-unit-tests-part-1-wrong-way.html

顺便说一句,如果您不使用 C# 5,请注意关闭 foreach 变量result请参阅

于 2013-05-21T16:56:32.223 回答
2

Microsoft.Bcl.Async我们无法向 Task 添加任何静态方法。但是,您可以找到大多数方法TaskEx,例如,TaskEx.WhenAll()确实存在。

于 2013-05-23T17:01:13.753 回答