任务<webresponse>。等待永远</webresponse>

时间:2012-09-21 06:41:59

标签: c# .net-4.5

C#,。Net 4.5。我有一个包含要处理的对象的队列。处理包括使用在对象的一个​​字段中指定的URL获得数据。在操作过程中,可以将新对象添加到队列中。当我尝试使用网络异步工作时,我遇到了一个问题。

这是最低限度的代码。

public partial class Form1 : Form
{
    private void button1_Click(object sender, EventArgs e)
    {
        string[] urls = { "http://www.stackoverflow.com/", 
                            "http://www.google.com/", 
                            "http://www.microsoft.com/" };
        int i = 0;
        Queue<MyClass1> queue = new Queue<MyClass1>();

        HttpWebRequest webRequest = (HttpWebRequest)WebRequest.Create(urls[i]);
        webRequest.Proxy.Credentials = CredentialCache.DefaultCredentials;
        queue.Enqueue(new MyClass1(urls[i], webRequest.GetResponseAsync()));

        while (queue.Count > 0)
        {
            MyClass1 o = queue.Dequeue();
            o.RespTask.Wait();
            Debug.Print("Url: {0}, bytes: {1}", o.Url, 
                o.RespTask.Result.ContentLength);

            i++;
            if (i < urls.Length)
            {
                webRequest = (HttpWebRequest)WebRequest.Create(urls[i]);
                webRequest.Proxy.Credentials = CredentialCache.DefaultCredentials;
                queue.Enqueue(new MyClass1(urls[i], webRequest.GetResponseAsync()));
            }
        } 
    }
}

public class MyClass1
{
    public MyClass1() { }
    public MyClass1(string url, Task<WebResponse> respTask)
    {
        Url = url;
        RespTask = respTask;
    }

    public string Url;
    public Task<WebResponse> RespTask;
}

该代码挂在o.RespTask.Wait();在循环的第三次迭代。在此调用之前o.RespTask.Status具有值WaitingForActivation并且等待持续。我做错了什么?

更新。我检查了3个盒子上的代码。其中两个(Win7 32位和Win7 64位)程序挂起。在第三个(Win7 64位)一切正常。在我看来非常奇怪。

1 个答案:

答案 0 :(得分:1)

此代码在此类修改后已停止挂起:

...
Debug.Print("Url: {0}, bytes: {1}", o.Url, 
    o.RespTask.Result.ContentLength);

o.RespTask.Result.Close();

i++;
...

我的错误是我没有注意到必须调用HttpWebResponse类的Close方法这一事实。