我们在开发web系统时有时会有以下需求:

  • 希望某类或者某已知MIME 类型的文件(比如:*.gif;*.txt;*.htm)能够在访问时弹出“文件下载”对话框
  • 希望以原始文件名(上传时的文件名,例如:山东省政府1024号文件.doc)提供下载,但服务器上保存的地址却是其他文件名(如:12519810948091234_asdf.doc)
  • 希望某文件直接在浏览器上显示而不是弹出文件下载对话框
近日在使用Content-disposition的使用出现UTF-8识别不了的情况,经过HttpUtility.UrlEncode(fileName,Encoding.UTF8)编码后的文件名称直接显示给用户了,如图:
Content-Disposition的使用和注意事项(转)_文件下载
名称为uft 8格式,但是ie6.0却识别不了,很是不解,以前曾经使用此种方法是可行的,只好利用Google了,经查找网络里大多为像Qihangnet写的这篇文章似的,我也贴出来供大家使用。
Qihangnet的这篇文章跟我的想法基本是相似的,在
HttpContext.Current.Response.AddHeader("Content-Disposition", "attachment; filename=\"" + UTF_FileName(filename) + ".doc\";"); .doc的后缀名是关键,我在没有写.doc或.txt之类的后缀时,依旧跟上图一样,加上后缀后utf 8格式的字符串自动识别成汉字了,至于为什么我还不是很清楚,贴出来让大家也注意一下就是了。

//------------------------------------------start-------------------------------------------------------------------

作者:Qihangnet 
出处:http://www.qihangnet.com/PermaLink,guid,db65d50a-ba90-4229-a3a2-71b4f1b407b9.aspx

 

    要解决上述需求就可以使用Content-disposition来解决。第一个需求的解决办法是

Response.AddHeader "content-disposition","attachment; filename=fname.ext"
 
将上述需求进行归我给出如下例子代码:
public static void ToDownload(string serverfilpath,string filename)
{
FileStream fileStream = new FileStream(serverfilpath, FileMode.Open);
long fileSize = fileStream.Length;
HttpContext.Current.Response.ContentType = "application/octet-stream";
HttpContext.Current.Response.AddHeader("Content-Disposition", "attachment; filename=\"" + UTF_FileName(filename) + ".doc\";");
////attachment --- 作为附件下载
////inline --- 在线打开
HttpContext.Current.Response.AddHeader("Content-Length", fileSize.ToString());
byte[] fileBuffer = new byte[fileSize];
fileStream.Read(fileBuffer, 0, (int)fileSize);
HttpContext.Current.Response.BinaryWrite(fileBuffer);
fileStream.Close();
HttpContext.Current.Response.End();
}
public static void ToOpen(string serverfilpath, string filename)
{
FileStream fileStream = new FileStream(serverfilpath, FileMode.Open);
long fileSize = fileStream.Length;
HttpContext.Current.Response.ContentType = "application/octet-stream";
HttpContext.Current.Response.AddHeader("Content-Disposition", "inline; filename=\"" + UTF_FileName(filename) + ".doc\";");
    HttpContext.Current.Response.AddHeader("Content-Length", fileSize.ToString());
byte[] fileBuffer = new byte[fileSize];
fileStream.Read(fileBuffer, 0, (int)fileSize);
HttpContext.Current.Response.BinaryWrite(fileBuffer);
fileStream.Close();
HttpContext.Current.Response.End();
}
private static string UTF_FileName(string filename)
{
return HttpUtility.UrlEncode(filename, System.Text.Encoding.UTF8);
}

 

简单的对上述代码做一下解析,ToDownload方法为将一个服务器上的文件(serverfilpath为服务器上的物理地址),以某文件名(filename)在浏览器上弹出“文件下载”对话框,而ToOpen是将服务器上的某文件以某文件名在浏览器中显示/打开的。注意其中我使用了UTF_FileName方法,该方法很简单,主要为了解决包含非英文/数字名称的问题,比如说文件名为“衣明志.doc”,使用该方法客户端就不会出现乱码了。
//*---------------------------end----------------------------------

园子里liping13599168的下载函数也不错,顺便也贴一下:
出处:javascript:void(0)

Content-Disposition的使用和注意事项(转)_文件下载_02Content-Disposition的使用和注意事项(转)_文件路径_03文件下载函数
using System.Threading;    
    
/// <summary>
    
/// 
    
/// </summary>
    
/// <param name="_Request">读取客户端在 Web 请求期间发送的 HTTP 值</param>
    
/// <param name="_Response">封装来自 ASP.NET 操作的 HTTP 响应信息</param>
    
/// <param name="_fileName">目的文件名称</param>
    
/// <param name="_fullPath">源文件路径</param>
    
/// <param name="_speed"></param>
    
/// <returns>是否成功</returns>
    public static bool ResponseFile(HttpRequest _Request, HttpResponse _Response, string _fileName, string _fullPath, long _speed)
    {
        
try
        {
            FileStream myFile 
= new FileStream(_fullPath, FileMode.Open, FileAccess.Read, FileShare.ReadWrite);
            BinaryReader br 
= new BinaryReader(myFile);
            
try
            {
                _Response.AddHeader(
"Accept-Ranges""bytes");
                _Response.Buffer 
= false;
                
long fileLength = myFile.Length;
                
long startBytes = 0;

                
double pack = 10240//10K bytes
                
//int sleep = 200;   //每秒5次   即5*10K bytes每秒
                int sleep = (int)Math.Floor(1000 * pack / _speed) + 1;
                
if (_Request.Headers["Range"!= null)
                {
                    _Response.StatusCode 
= 206;
                    
string[] range = _Request.Headers["Range"].Split(new char[] { '=''-' });
                    startBytes 
= Convert.ToInt64(range[1]);
                }
                _Response.AddHeader(
"Content-Length", (fileLength - startBytes).ToString());
                
if (startBytes != 0)
                {
                    
//Response.AddHeader("Content-Range", string.Format(" bytes {0}-{1}/{2}", startBytes, fileLength-1, fileLength));
                }
                _Response.AddHeader(
"Connection""Keep-Alive");
                _Response.ContentType 
= "application/octet-stream";
                _Response.AppendHeader(
"Content-Disposition""attachment; filename=\"" + HttpUtility.UrlEncode(_fileName, System.Text.Encoding.UTF8) + ".txt\";");

                br.BaseStream.Seek(startBytes, SeekOrigin.Begin);
                
int maxCount = (int)Math.Floor((fileLength - startBytes) / pack) + 1;

                
for (int i = 0; i < maxCount; i++)
                {
                    
if (_Response.IsClientConnected)
                    {
                        _Response.BinaryWrite(br.ReadBytes(
int.Parse(pack.ToString())));
                        Thread.Sleep(sleep);
                    }
                    
else
                    {
                        i 
= maxCount;
                    }
                }
            }
            
catch
            {
                
return false;
            }
            
finally
            {
                br.Close();

                myFile.Close();
            }
        }
        
catch
        {
            
return false;
        }
        
return true;
}

//调用方法
Page.Response.Clear();
bool success = ResponseFile(Page.Request, Page.Response, "目的文件名称"@"源文件路径"1024000);
if (!success)
     Response.Write(
"下载文件出错!");
Page.Response.End();