Почему мой StreamResult не возвращает целые файлы?
Я должен обслуживать большие файлы (200-800 МБ) клиенту с моего контроллера. Я протестировал FileStreamResult, но этот класс буферизовал весь файл в памяти. Такое поведение недостаточно для моего проекта.
Далее я тестирую подход отсюда: http://support.microsoft.com/kb/812406. Что касается памяти, это выглядит довольно хорошо, но файлы загружаются не полностью на клиенте (исходный файл - 210222 КБ, загруженные файлы - с 209551 по 209776). Это означает, что потеряно около 0,5 МБ (что приводит к повреждению файлов).
У кого-нибудь есть идея? Какой лучший способ сделать это в любом случае? Я благодарен за все.
Просто для пользователей в будущем ссылка указывала на следующий код:
System.IO.Stream iStream = null;
// Buffer to read 10K bytes in chunk:
byte[] buffer = new Byte[10000];
// Length of the file:
int length;
// Total bytes to read:
long dataToRead;
// Identify the file to download including its path.
string filepath = "DownloadFileName";
// Identify the file name.
string filename = System.IO.Path.GetFileName(filepath);
try
{
// Open the file.
iStream = new System.IO.FileStream(filepath, System.IO.FileMode.Open,
System.IO.FileAccess.Read,System.IO.FileShare.Read);
// Total bytes to read:
dataToRead = iStream.Length;
Response.ContentType = "application/octet-stream";
Response.AddHeader("Content-Disposition", "attachment; filename=" + filename);
// Read the bytes.
while (dataToRead > 0)
{
// Verify that the client is connected.
if (Response.IsClientConnected)
{
// Read the data in buffer.
length = iStream.Read(buffer, 0, 10000);
// Write the data to the current output stream.
Response.OutputStream.Write(buffer, 0, length);
// Flush the data to the HTML output.
Response.Flush();
buffer= new Byte[10000];
dataToRead = dataToRead - length;
}
else
{
//prevent infinite loop if user disconnects
dataToRead = -1;
}
}
}
catch (Exception ex)
{
// Trap the error, if any.
Response.Write("Error : " + ex.Message);
}
finally
{
if (iStream != null)
{
//Close the file.
iStream.Close();
}
Response.Close();
}
Обновить
Это мое действие:
public DownloadResult TransferTest()
{
string fullFilePath = @"C:\ws\Test\Test\Templates\example.pdf";
return new DownloadResult(fullFilePath);
}
Я просто вызываю действие прямо из моего браузера ( http://xxx.xxx/Other/TransferTest).
1 ответ
Код в основном выглядит нормально - вы более или менее правильно обрабатываете возвращаемое значение из Read
(если бы я был разборчив, я бы сказал, проверить это для <=0
, но это не будет ожидаемым поведением, так как вы, вероятно, заблокировали файл).
Единственное, что происходит: попробуйте добавить:
Response.OutputStream.Flush();
и, возможно:
Response.OutputStream.Close();
чтобы убедиться, что выходной поток сброшен.