使用FileResult在Asp.Net MVC中下载任何类型的文件?

时间:2010-08-30 22:00:53

标签: c# asp.net-mvc-2

我已经向我建议我应该使用FileResult来允许用户从我的Asp.Net MVC应用程序下载文件。但我能找到的唯一例子总是与图像文件有关(指定内容类型image / jpeg)。

但是,如果我不知道文件类型怎么办?我希望用户能够从我网站的文件区下载几乎任何文件。

我已经阅读了一个这样做的方法(请参阅代码previous post),实际上工作正常,除了一件事:“另存为”对话框中出现的文件的名称是连接起来的带下划线的文件路径(folder_folder_file.ext)。此外,似乎人们认为我应该返回一个FileResult,而不是使用我找到BinaryContentResult的自定义类。

任何人都知道在MVC中进行此类下载的“正确”方法吗?

编辑: 我得到了答案(下面),但我想如果其他人有兴趣我应该发布完整的工作代码:

public ActionResult Download(string filePath, string fileName)
{
    string fullName = Path.Combine(GetBaseDir(), filePath, fileName);

    byte[] fileBytes = GetFile(fullName);
    return File(
        fileBytes, System.Net.Mime.MediaTypeNames.Application.Octet, fileName);
}

byte[] GetFile(string s)
{
    System.IO.FileStream fs = System.IO.File.OpenRead(s);
    byte[] data = new byte[fs.Length];
    int br = fs.Read(data, 0, data.Length);
    if (br != fs.Length)
        throw new System.IO.IOException(s);
    return data;
}

9 个答案:

答案 0 :(得分:388)

您只需指定通用八位字节流MIME类型:

public FileResult Download()
{
    byte[] fileBytes = System.IO.File.ReadAllBytes(@"c:\folder\myfile.ext");
    string fileName = "myfile.ext";
    return File(fileBytes, System.Net.Mime.MediaTypeNames.Application.Octet, fileName);
}

答案 1 :(得分:100)

MVC框架本身支持这一点。 System.Web.MVC.Controller.File控制器提供了按name / stream / array返回文件的方法。

例如,使用文件的虚拟路径,您可以执行以下操作。

return File(virtualFilePath, System.Net.Mime.MediaTypeNames.Application.Octet,  Path.GetFileName(virtualFilePath));

答案 2 :(得分:32)

如果您使用的是.NET Framework 4.5,那么您可以使用MimeMapping.GetMimeMapping(字符串FileName)来获取文件的MIME类型。这就是我在行动中使用它的方式。

return File(Path.Combine(@"c:\path", fileFromDB.FileNameOnDisk), MimeMapping.GetMimeMapping(fileFromDB.FileName), fileFromDB.FileName);

答案 3 :(得分:10)

Phil Haack有一个很好的article,他在那里创建了一个Custome File Download Action Result类。您只需指定文件的虚拟路径和要保存为的名称。

我用了一次,这是我的代码。

        [AcceptVerbs(HttpVerbs.Get)]
        public ActionResult Download(int fileID)
        {
            Data.LinqToSql.File file = _fileService.GetByID(fileID);

            return new DownloadResult { VirtualPath = GetVirtualPath(file.Path),
                                        FileDownloadName = file.Name };
        }

在我的例子中,我存储了文件的物理路径,所以我使用了这个帮助方法 - 我找到了一个我记不起来的地方 - 将它转换为虚拟路径

        private string GetVirtualPath(string physicalPath)
        {
            string rootpath = Server.MapPath("~/");

            physicalPath = physicalPath.Replace(rootpath, "");
            physicalPath = physicalPath.Replace("\\", "/");

            return "~/" + physicalPath;
        }

这是完整的课程,取自Phill Haack的文章

public class DownloadResult : ActionResult {

    public DownloadResult() {}

    public DownloadResult(string virtualPath) {
        this.VirtualPath = virtualPath;
    }

    public string VirtualPath {
        get;
        set;
    }

    public string FileDownloadName {
        get;
        set;
    }

    public override void ExecuteResult(ControllerContext context) {
        if (!String.IsNullOrEmpty(FileDownloadName)) {
            context.HttpContext.Response.AddHeader("content-disposition", 
            "attachment; filename=" + this.FileDownloadName)
        }

        string filePath = context.HttpContext.Server.MapPath(this.VirtualPath);
        context.HttpContext.Response.TransmitFile(filePath);
    }
}

答案 4 :(得分:5)

感谢 Ian Henry

如果您需要 从MS SQL Server获取文件 ,这是解决方案。

public FileResult DownloadDocument(string id)
        {
            if (!string.IsNullOrEmpty(id))
            {
                try
                {
                    var fileId = Guid.Parse(id);

                    var myFile = AppModel.MyFiles.SingleOrDefault(x => x.Id == fileId);

                    if (myFile != null)
                    {
                        byte[] fileBytes = myFile.FileData;
                        return File(fileBytes, System.Net.Mime.MediaTypeNames.Application.Octet, myFile.FileName);
                    }
                }
                catch
                {
                }
            }

            return null;
        }

AppModel EntityFramework型号, MyFiles 在数据库中显示文件数据 MyFiles 表格中为varbinary(MAX)

答案 5 :(得分:2)

它很简单,只需在文件名为

的directoryPath中提供物理路径即可
public FilePathResult GetFileFromDisk(string fileName)
{
    return File(directoryPath, "multipart/form-data", fileName);
}

答案 6 :(得分:0)

   public ActionResult Download()
        {
            var document = //Obtain document from database context
    var cd = new System.Net.Mime.ContentDisposition
    {
        FileName = document.FileName,
        Inline = false,
    };
            Response.AppendHeader("Content-Disposition", cd.ToString());
            return File(document.Data, document.ContentType);
        }

答案 7 :(得分:-1)

如果(string.IsNullOrWhiteSpace(fileName))                 返回Content(“文件名不存在”);

        var path = Path.Combine(your path, your filename);

        var stream = new FileStream(path, FileMode.Open);

        return File(stream, System.Net.Mime.MediaTypeNames.Application.Octet, fileName);

答案 8 :(得分:-4)

GetFile应该关闭文件(或在使用中打开它)。然后,您可以在转换为字节后删除文件 - 下载将在该字节缓冲区上完成。

    byte[] GetFile(string s)
    {
        byte[] data;
        using (System.IO.FileStream fs = System.IO.File.OpenRead(s))
        {
            data = new byte[fs.Length];
            int br = fs.Read(data, 0, data.Length);
            if (br != fs.Length)
                throw new System.IO.IOException(s);
        }
        return data;
    }

所以在你的下载方法......

        byte[] fileBytes = GetFile(file);
        // delete the file after conversion to bytes
        System.IO.File.Delete(file);
        // have the file download dialog only display the base name of the file            return File(fileBytes, System.Net.Mime.MediaTypeNames.Application.Octet, Path.GetFileName(file));