kiedyś klasy pochodzący z FileResult
osiągnięcie tego za pomocą normalnego MVC wzór:
/// <summary>
/// MVC action result that generates the file content using a delegate that writes the content directly to the output stream.
/// </summary>
public class FileGeneratingResult : FileResult
{
/// <summary>
/// The delegate that will generate the file content.
/// </summary>
private readonly Action<System.IO.Stream> content;
private readonly bool bufferOutput;
/// <summary>
/// Initializes a new instance of the <see cref="FileGeneratingResult" /> class.
/// </summary>
/// <param name="fileName">Name of the file.</param>
/// <param name="contentType">Type of the content.</param>
/// <param name="content">Delegate with Stream parameter. This is the stream to which content should be written.</param>
/// <param name="bufferOutput">use output buffering. Set to false for large files to prevent OutOfMemoryException.</param>
public FileGeneratingResult(string fileName, string contentType, Action<System.IO.Stream> content,bool bufferOutput=true)
: base(contentType)
{
if (content == null)
throw new ArgumentNullException("content");
this.content = content;
this.bufferOutput = bufferOutput;
FileDownloadName = fileName;
}
/// <summary>
/// Writes the file to the response.
/// </summary>
/// <param name="response">The response object.</param>
protected override void WriteFile(System.Web.HttpResponseBase response)
{
response.Buffer = bufferOutput;
content(response.OutputStream);
}
}
Sposób kontroler będzie teraz następująco:
public ActionResult Export(int id)
{
return new FileGeneratingResult(id + ".csv", "text/csv",
stream => this.GenerateExportFile(id, stream));
}
public void GenerateExportFile(int id, Stream stream)
{
stream.Write(/**/);
}
Należy pamiętać, że jeśli buforowanie jest wyłączone,
stream.Write(/**/);
staje się bardzo wolny. Rozwiązaniem jest użycie BufferedStream. Zwiększenie wydajności o około 100x w jednym przypadku. Zobacz
Unbuffered Output Very Slow
Powinieneś unikać Response.End() http://stevesmithblog.com/blog/use-httpapplication-completerequest-instead-response-end/ –
Zaktualizowany następnie do użycia CompleteRequest(). – womp
Może być użyteczne zastąpienie "return View()" przez "return Content (" ")", aby uniknąć błędów związanych z brakującym widokiem. Ale czy to podejście jest bezpieczne? –