在ASP.NET中流式传输文件的最佳方式

时间:2009-03-03 22:22:43

标签: asp.net flash streaming video-streaming

使用ASP.NET流式传输文件的最佳方法是什么?

似乎有各种方法,我目前在http处理程序中使用Response.TransmitFile()方法,该方法直接将文件发送到浏览器。这用于各种事情,包括从Webroot外部发送FLV到嵌入式Flash视频播放器。

但是,这似乎不是一种可靠的方法。特别是, Internet Explorer(7)存在一个奇怪的问题,浏览器只是在观看一两个视频后挂起。点击任何链接等都没有任何效果,在网站上再次运行的唯一方法是关闭浏览器并重新打开它。

在其他浏览器中也会出现这种情况,但频率较低。基于一些基本的测试,我怀疑这与文件流的方式有关...也许连接没有正确关闭,或者沿着这些线路。

在尝试了几件不同的事情之后,我发现以下方法对我有用:

Response.WriteFile(path);
Response.Flush();
Response.Close();
Response.End();

这解决了上述问题,观看视频不再导致Internet Explorer挂起。

但是,我的理解是Response.WriteFile()首先将文件加载到内存中,并且考虑到流式传输的某些文件可能非常大,这似乎不是一个理想的解决方案。

我很想知道其他开发人员如何在ASP.NET中传输大型文件,特别是流式传输FLV视频文件。

4 个答案:

答案 0 :(得分:51)

我会把事情放在“aspx”管道之外。特别是,我会写一个运行处理程序(ashx,或通过配置映射),它执行最小工作,并简单地以块的形式写入响应。处理程序将正常接受来自查询字符串/表单的输入,定位要流的对象,并流式传输数据(在循环中使用中等大小的本地缓冲区)。一个简单(不完整)的例子如下所示:

public void ProcessRequest(HttpContext context) {
    // read input etx
    context.Response.Buffer = false;
    context.Response.ContentType = "text/plain";
    string path = @"c:\somefile.txt";
    FileInfo file = new FileInfo(path);
    int len = (int)file.Length, bytes;
    context.Response.AppendHeader("content-length", len.ToString());
    byte[] buffer = new byte[1024];
    Stream outStream = context.Response.OutputStream;
    using(Stream stream = File.OpenRead(path)) {
        while (len > 0 && (bytes =
            stream.Read(buffer, 0, buffer.Length)) > 0)
        {
            outStream.Write(buffer, 0, bytes);
            len -= bytes;
        }
    }
}

答案 1 :(得分:10)

看看下面的文章Tracking and Resuming Large File Downloads in ASP.NET,它将为您提供更深入的内容,而不仅仅是打开一个流并清除掉所有内容。

http协议支持远程字节请求和可恢复下载,许多流媒体客户端(如视频播放器或Adobe pdf)可以并且将尝试将这些内容组合起来,从而节省带宽并为用户提供更好的体验。

不是微不足道的,但现在是时候花光了。

答案 2 :(得分:8)

尝试将文件作为流打开,然后使用Response.OutputStream.Write()。例如:

编辑:我的不好,我忘记了Write需要一个字节缓冲区。固定

byte [] buffer = new byte[1<<16] // 64kb
int bytesRead = 0;
using(var file = File.OpenRead(path))
{
   while((bytesRead = file.Read(buffer, 0, buffer.Length)) != 0)
   {
        Response.OutputStream.Write(buffer, 0, bytesRead);
   }
}
Response.Flush();
Response.Close();
Response.End();

编辑2:你试过这个吗?它应该工作。

答案 3 :(得分:3)

在尝试了许多不同的组合后,包括在各种答案中发布的代码,似乎在调用TransmitFile之前设置了Response.Buffer = true,并且Web应用程序现在在Internet Explorer中响应速度更快。

在这种特殊情况下,SWF扩展也映射到ASP.NET,我们在Web应用程序中使用自定义处理程序从磁盘读取文件,然后使用Response.TransmitFile()将它们发送到浏览器。我们有一个基于闪存的视频播放器来播放也是SWF的视频文件,我认为让所有这些活动通过处理程序而不进行缓冲是可能导致IE中发生奇怪事情的原因。