建议使用文件扩展名创建ActionResult的方法

Sim*_*ver 19 asp.net-mvc actionresult

我需要在ASP.NET MVC应用程序中创建一个具有.csv文件类型的ActionResult.

我将向我的营销合作伙伴提供"不要打电话"的电子邮件列表,我希望它在文件类型中具有.csv扩展名.然后它将自动在Excel中打开.

 http://www.example.com/mailinglist/donotemaillist.csv?password=12334
Run Code Online (Sandbox Code Playgroud)

我已成功完成以下操作,但我想确保这是绝对最佳和推荐的方法.

    [ActionName("DoNotEmailList.csv")]
    public ContentResult DoNotEmailList(string username, string password)
    {
            return new ContentResult()
            {
                Content = Emails.Aggregate((a,b)=>a+Environment.NewLine + b), 
                ContentType = "text/csv"
            };
    }
Run Code Online (Sandbox Code Playgroud)

这个Actionmethod会响应上面的链接就好了.

我只是想知道是否有任何意外冲突的任何不同版本的IIS,任何类型的ISAPI过滤器,或任何我现在无法想到的文件扩展名.

我需要100%肯定,因为我会向外部合作伙伴提供这个,并且不想在以后改变主意.我真的看不出任何问题,但也许有些东西模糊不清 - 或者更像"MVC"这样做的方式.

Pet*_*esz 29

我使用FileContentResult操作也做了类似的事情.

public FileContentResult DoNotEmailList(string username, string password)
{
        string csv = Emails.Aggregate((a,b)=>a+Environment.NewLine + b);
        byte[] csvBytes = ASCIIEncoding.ASCII.GetBytes( csv );
        return File(csvBytes, "text/csv", "DoNotEmailList.csv");
}
Run Code Online (Sandbox Code Playgroud)

它将为您添加内容处置标头.


eu-*_*-ne 21

我认为你的Response必须包含"Content-Disposition"标题.像这样创建自定义ActionResult:

public class MyCsvResult : ActionResult {

    public string Content {
        get;
        set;
    }

    public Encoding ContentEncoding {
        get;
        set;
    }

    public string Name {
        get;
        set;
    }

    public override void ExecuteResult(ControllerContext context) {
        if (context == null) {
            throw new ArgumentNullException("context");
        }

        HttpResponseBase response = context.HttpContext.Response;

        response.ContentType = "text/csv";

        if (ContentEncoding != null) {
            response.ContentEncoding = ContentEncoding;
        }

        var fileName = "file.csv";

        if(!String.IsNullOrEmpty(Name)) {
            fileName = Name.Contains('.') ? Name : Name + ".csv";
        }

        response.AddHeader("Content-Disposition",
            String.Format("attachment; filename={0}", fileName));

        if (Content != null) {
            response.Write(Content);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

并在Action而不是ContentResult中使用它:

return new MyCsvResult {
    Content = Emails.Aggregate((a,b) => a + Environment.NewLine + b)
    /* Optional
     * , ContentEncoding = ""
     * , Name = "DoNotEmailList.csv"
     */
};
Run Code Online (Sandbox Code Playgroud)

  • 运行此代码时出错(MVC2,ASP.Net 3.5):"此操作需要IIS集成管道模式".看起来需要更改response.Headers.Add to response.AddHeader - http://www.platinumbay.com/blogs/dotneticated/archive/2010/10/07/this-operation-requires-iis-integrated-pipeline-mode的.aspx (5认同)
  • 我发现了bug(`Response.Headers.Add ...`) - 修复了(应该是`response.Headers.Add ...`) - 再次感谢AD. (2认同)

Tal*_*joe 6

这就是我做类似的事情.我将其视为下载:

var disposition = String.Format(
  "attachment;filename=\"{0}.csv\"", this.Model.Name);
Response.AddHeader("content-disposition", disposition);
Run Code Online (Sandbox Code Playgroud)

这应该在浏览器中显示为具有给定文件名的文件下载.

但是,我想不出你为什么不能工作的原因.