我正在使用 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 连接时,请求被拆分成一堆块,并多次调用 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/