我正在创建一个.Net Web API应用程序,其中以下代码调用我的不同c#应用程序来下载文件,然后将其保存在磁盘上。有时一切正常,我得到了文件,但有时下面的代码无法读取流,我可以看到远程连接在我的其他应用程序中关闭异常。
public async Task<string> GetFilePathAsync(TestModel model)
{
string filePath = string.Empty;
var response = await cgRequestHelper.DownloadAsync(model);
if (response.IsSuccessStatusCode)
{
filePath = await SaveCgStreamAsync(cgResponse, serviceModel.FileName);
}
return filePath;
}
public async Task<HttpResponseMessage> DownloadAsync(TestModel model)
{
if (model == null)
throw new ArgumentNullException("model");
if (string.IsNullOrEmpty(model.Url))
throw new ArgumentNullException("Url");
if (model.Headers == null)
throw new ArgumentNullException("Headers");
HttpResponseMessage response;
using (HttpClient httpClient = new HttpClient())
{
foreach (var header in model.Headers)
{
httpClient.DefaultRequestHeaders.Add(header.Key, header.Value);
}
response = await httpClient.GetAsync(model.Url, HttpCompletionOption.ResponseHeadersRead);
}
return response;
}
public async Task<string> SaveCgStreamAsync(HttpResponseMessage response, string fileName)
{
if (response == null)
throw new ArgumentNullException("response");
if (string.IsNullOrEmpty(fileName))
throw new ArgumentNullException("fileName");
var filePath = _CreateTemporaryLocation(fileName);
Stream cgStream = null;
Stream fileStream = null;
try
{
cgStream = await response.Content.ReadAsStreamAsync();
fileStream = File.Open(filePath, FileMode.Create);
await cgStream.CopyToAsync(fileStream);
}
catch(Exception e)
{
throw;
}
finally
{
if(cgStream != null)
cgStream.Dispose();
if(fileStream != null)
fileStream.Dispose();
}
return filePath;
}
我在Global.asax.cs中设置了ServicePointManager.DefaultConnectionLimit = 1000
在我当前的应用程序中,我在“await cgStream.CopyToAsync(fileStream);”上遇到异常。尝试读取cgStream时的行。唯一的例外是“无法访问已关闭的流”。使用null InnerException 另一个应用程序例外是: System.Web.HttpException:远程主机关闭了连接。错误代码是0x800703E3。 在System.Web.Hosting.IIS7WorkerRequest.RaiseCommunicationError(Int32 result,Boolean throwOnDisconnect) 在System.Web.Hosting.IIS7WorkerRequest.ExplicitFlush() 在System.Web.HttpResponse.Flush(Boolean finalFlush,Boolean async)
平均而言,10个请求中有1个因上述错误而失败。由于它是随机的并且并非总是失败,因此很难解决问题。
上述代码是在http://www.tugberkugurlu.com/archive/efficiently-streaming-large-http-responses-with-httpclient
的帮助下编写的感谢任何与此问题相关的帮助!
由于
答案 0 :(得分:3)
我在代码中找出了问题: 问题是我在'使用'中初始化HttpClient对象并在使用范围之外使用它的响应。这将处置HttpClient对象,从而打破远程连接,这就是我无法流式传输内容的原因。随机行为是因为我们不知道对象什么时候会被处理掉,有时它在流式传输之前不会被处理掉,有时它会被处理掉。
答案 1 :(得分:0)
我会尝试Stream.CopyTo
而只是为了让您知道我之前尝试过以下代码,后来将其更改为CopyToAsync
public async Task<string> GetFilePathAsync(TestModel model)
{
string filePath = string.Empty;
var response = await cgRequestHelper.DownloadAsync(model);
if (response.IsSuccessStatusCode)
{
Stream cgStream = await response.Content.ReadAsStreamAsync();
filePath = SaveCgStream(cgStream , model.FileName);
}
return filePath;
}
public string SaveCgStream(Stream cgStream, string fileName)
{
if (response == null)
throw new ArgumentNullException("response");
if (string.IsNullOrEmpty(fileName))
throw new ArgumentNullException("fileName");
const int CHUNK_SIZE = 40000;
var filePath = _CreateTemporaryLocation(fileName);
var fileStream = File.Create(filePath, CHUNK_SIZE);
int bytesRead;
var buffer = new byte[CHUNK_SIZE];
bool isReadStarted = false;
try
{
do
{
bytesRead = cgStream.Read(buffer, 0, CHUNK_SIZE);
if (bytesRead > 0)
{
isReadStarted = true;
fileStream.Write(buffer, 0, bytesRead);
}
} while (bytesRead > 0);
}
catch (Exception e)
{
throw e;
}
finally
{
fileStream.Close();
}
return filePath;
}
我在bytesRead = cgStream.Read(buffer, 0, CHUNK_SIZE);
行中遇到了同样的错误。你还认为这是CopyToAsync
问题吗?
我对此表示怀疑,因为当我提出一个断点时,我可以看到该流是不可读的,因为Read属性在进入CopyToAsync
方法之前是假的。
我尝试了CopyTo
而不是CopyToAsync
,但没有帮助。截图:
答案 2 :(得分:-1)
根据文档,Stream.CopyToAsync
是异步的。这意味着它不会阻止调用者(您的方法),这将继续在finally块中处理流。换句话说,你有竞争条件。切换到Stream.CopyTo
你应该很高兴。