0

我必须从我的控制器向客户端提供大文件(200-800MB)。我测试了 FileStreamResult,但是这个类在内存中缓冲了整个文件。这种行为对我的项目来说不够好。

我进一步测试了这里的方法:http: //support.microsoft.com/kb/812406。关于内存,这看起来不错——但是文件没有完全下载到客户端(原始文件是 210222 KB,下载的文件是 209551 到 209776)。这意味着大约丢失了 0.5 MB(这最终导致文件被破坏)。

有人有想法吗?无论如何,最好的方法是什么?我感谢一切。

只是为了将来的用户,链接指向以下代码:

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 ) 调用该操作。

4

1 回答 1

3

该代码基本上看起来不错-您或多或少正确处理了来自的返回值Read(如果我很挑剔,我会说检查它<=0,但这不是预期的行为,因为您可能锁定了文件)。

唯一发生的是:尝试添加:

Response.OutputStream.Flush();

也许:

Response.OutputStream.Close();

确保刷新输出流。

于 2014-04-17T14:57:06.673 回答