c# - HttpWebRequest 对于分块数据很慢

标签 c# .net http web

我正在使用 HttpWebRequest 连接到我内部构建的 HTTP 服务器。我的问题是它比通过例如 PostMan ( https://chrome.google.com/webstore/detail/postman-rest-client/fdmmgilgnpjigdojojpjoooidkmcomcm?hl=en ) 连接到服务器慢很多,后者可能使用 Chrome 中的内置函数来请求数据。

服务器是使用 MSDN ( http://msdn.microsoft.com/en-us/library/dxkwh6zw.aspx ) 上的这个示例构建的,并使用大小为 64 的缓冲区。该请求是一个 HTTP 请求,正文中包含一些数据。

当通过 PostMan 连接时,请求被分成一堆 block 并且 BeginRecieve() 被调用多次,每次接收 64B 并花费大约 2 毫秒。除了最后一个,它接收不到 64B。

但是当使用 HttpWebRequest 连接到我的客户端时,第一个 BeginRecieve() 回调接收到 64B 并花费了大约 1 毫秒,接下来仅接收到 47B 并花费了将近 200 毫秒,最后第三个接收到大约 58B 并花费了 2 毫秒。

第二个 BeginRecieve 是怎么回事?我注意到,一旦我开始将数据写入 HttpWebRequest 输入流,就会建立连接,但在我调用 GetResponse() 之前数据接收不会开始。

这是我的 HttpWebRequest 代码:

var request = (HttpWebRequest)WebRequest.Create(url);

request.Method = verb;
request.Timeout = timeout;
request.Proxy = null;
request.KeepAlive = false;
request.Headers.Add("Content-Encoding", "UTF-8");
System.Net.ServicePointManager.Expect100Continue = false;
request.ServicePoint.Expect100Continue = false;

if ((verb == "POST" || verb == "PUT") && !String.IsNullOrEmpty(data))
{
    var dataBytes = Encoding.UTF8.GetBytes(data);

    try
    {
        var dataStream = request.GetRequestStream();
        dataStream.Write(dataBytes, 0, dataBytes.Length);
        dataStream.Close();
    }
    catch (Exception ex)
    {
        throw;
    }

}

WebResponse response = null;
try
{
    response = request.GetResponse();
}
catch (Exception ex)
{
    throw;
}

var responseReader = new StreamReader(rStream, Encoding.UTF8);
var responseStr = responseReader.ReadToEnd();

responseReader.Close();
response.Close();

我做错了什么?为什么它的行为与来自 Web 浏览器的 HTTP 请求如此不同?这实际上为我的应用程序增加了 200 毫秒的延迟。

最佳答案

这看起来像是 Nagle algorithm 的典型案例与TCP Delayed Acknowledgement发生冲突.在您的情况下,您正在发送一个小的 Http 请求(根据您的数字约为 170 个字节)。这可能小于 MSS(最大段大小),这意味着 Nagle 算法将启动。服务器可能会延迟 ACK,导致最多 500 毫秒的延迟。有关详细信息,请参阅链接。

您可以通过 ServicePointManager.UseNagleAlgorithm = false 禁用 Nagle(在发出第一个请求之前),参见 MSDN .

另见 Nagle’s Algorithm is Not Friendly towards Small Requests详细讨论,包括 Wireshark 分析。

注意:在您的回答中,当您进行写入-写入-读取时,您会遇到同样的情况。当你切换到写-读时,你就克服了这个问题。但是,我不相信您可以指示 HttpWebRequest(或 HttpClient)将小请求作为单个 TCP 写入操作发送。在某些情况下,这可能是一个很好的优化。虽然它可能会导致一些额外的数组复制,从而对性能产生负面影响。

关于c# - HttpWebRequest 对于分块数据很慢,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18465504/

相关文章:

c# - Xamarin Forms Prism 调用方法

c# - .net lambda 表达式和输出参数

c# - 我可以将多个值传递给 C# 中的 Setter 吗?

c# - 有哪些适合练习的 C# 问题?

c# - 我可以在没有泛型的情况下定义/约束成员实现两个接口(interface)吗?

c# - 泛型与 IL?

http - 如何使用 Java EE 的 ReSTLet api 从 ServerResource 发送 HTTP Get 请求?

java - 如何让 clojure 向套接字写入响应?

javascript - 为什么我的页面标题中有不间断空格?

c# - 这些静态变量的可能替代方案是什么?