【发布时间】:2014-03-25 03:41:30
【问题描述】:
我正在输入一些调试代码来测试一些东西,然后调试代码没有按预期运行。下面的示例是演示我的问题的简化代码。
这是在 .NET 4 中并使用 WebApi,我试图在调试代码中打印出 http 请求的正文。为此,我寻找输入流并读取该流。第一次运行良好,但如果我再次尝试读取它,我会得到一个空字符串。
为什么我不能第二次回溯并读取 InputStream?在下面的示例中,body2 始终为空。在第二组中,CanSeek 仍然为真,第二次调用 ReadToEnd() 返回一个空字符串,覆盖默认值。
using System.IO;
using System.Net;
using System.Net.Http;
using System.Web;
using System.Web.Http;
public class TestController : ApiController
{
public class TestOutuput
{
public string firstRead;
public string secondRead;
}
public HttpResponseMessage Post()
{
string body1 = "default for one";
string body2 = "default for two";
if (HttpContext.Current.Request.InputStream.CanSeek)
{
HttpContext.Current.Request.InputStream.Seek(0, System.IO.SeekOrigin.Begin);
}
using (var reader = new StreamReader(HttpContext.Current.Request.InputStream))
{
body1 = reader.ReadToEnd();
}
if (HttpContext.Current.Request.InputStream.CanSeek)
{
HttpContext.Current.Request.InputStream.Seek(0, System.IO.SeekOrigin.Begin);
}
using (var reader2 = new StreamReader(HttpContext.Current.Request.InputStream))
{
// this is always empty, even after seek back to origin
body2 = reader2.ReadToEnd();
}
TestOutuput testOutput = new TestOutuput() { firstRead = body1, secondRead = body2 };
HttpResponseMessage response = new HttpResponseMessage();
return Request.CreateResponse<TestOutuput>(HttpStatusCode.OK, testOutput);
}
}
【问题讨论】:
-
CanSeek是否返回true?您目前正在有条件地寻找,但随后假设寻找已经奏效。你为什么还要这样做?我个人期望至少对于大型请求主体,流可能不会在内存中缓冲。如果您想创建副本,我建议您首先然后从您的副本中多次阅读。将 Stream.CopyTo 与 MemoryStream 结合使用。 -
是的,CanSeek 确实返回 true 并执行 ReadToEnd。好点子。我将更新问题以澄清。
-
你试过 Fiddler 吗?
-
你可以像 Jon 指出的那样将流复制到 MemoryStream,但是 Web 请求中的 Stream 接口实际上只是不支持 all的 TCP 流的抽象> 到您可以在该套接字上执行的可能的 TCP 操作(即,您可以请求再次发送数据包——但如果您只是想重新读取数据,您很少想要这样做) .
-
@JonSkeet - 所有的好建议,谢谢。
标签: c# .net asp.net-mvc asp.net-web-api