2011-09-27 42 views
15

是否可以使用HTTP POST下载文件? 我知道“获取”方式(windows.location),但在我的情况下,有很多参数应该传递给服务器是否可以使用HTTP POST下载文件?

+0

这是POST的优点,可以发送大量数据(aka payload)。安全也在那里,但在你的情况下,发送参数是要求。 – EMM

回答

14

是的,POST请求的其余部分可以指示浏览器下载文件。文件内容将作为HTTP响应发送,与GET情况相同。

3

有没有区别,不是请求方法等,以及如何将数据发送到服务器。无论您使用GET还是POST,处理响应的方式都是相同的。

+0

嗨,格雷格,你能举个例子吗?非常感谢 – Sean

4

从某种意义上说,每一个HTTP GET或POST的“下载文件”,但最好把它作为消息有效载荷,而不是文件。在大多数情况下,有效负载是浏览器应呈现为网页的HTML文档。但是,如果它不是一个HTML文档呢?如果它是一个浏览器应该为用户提供“另存为”对话框的zip文件呢?显然,浏览器必须确定响应的内容类型并正确处理。

之一,一个浏览器确定该内容类型的最常用的方法是通过所谓的,因此HTTP header,“内容类型”。该头文件采用MIME类型的值。这是浏览器做内容特定的事情的关键,如响应中包含PDF文件等时启动acrobat插件等。

请注意,并非所有浏览器1)以相同的方式确定内容类型,并且2)反应以相同的方式转换为内容类型。有时你必须设置标题来玩弄所有浏览器所需的行为。所有服务器端技术都允许您设置HTTP标头。

12

看起来你想从Javascript生成POST请求。我相信没有办法让浏览器将AJAX请求的结果视为下载。即使Content-Type设置为浏览器通常提供的下载内容(例如“application/octet-stream”),浏览器也只会将数据存放在XMLHttpRequest对象中。

而且,正如你可能已经知道,有没有办法让window.open()发出一个POST请求。

我认为最好的办法是使生成的服务器上的文件的AJAX请求。在浏览器上,当请求完成时,使用window.open()来下载生成的文件。

+0

这是可能的解决方案之一,另一个是做表单提交检索适当的文件。 –

11

您的意思是这样的?

function IssuePostRequest(objData) 
    { 
     var strPageURL = "about:blank"; 
     var strAction = "@Url.Action("GetPDF", "Home")/"; 
     //var strAction = "/popups/delete.aspx"; 

     var strWindowName = "MyEvilHttpPostInAnewWindow"; // ifrmDownload 
     var iWindowWidth = 805; 
     var iWindowHeight = 625; 



     var form = document.createElement("form"); 
     form.setAttribute("id", "bla"); 
     form.setAttribute("method", "post"); 
     form.setAttribute("action", strAction); 
     form.setAttribute("target", strWindowName); 
     form.setAttribute("style", "display: none;"); 
     // setting form target to a window named 'formresult' 


     // Repeat for all data fields 
     var hiddenField = document.createElement("input"); 
     hiddenField.setAttribute("name", "data"); 
     hiddenField.setAttribute("value", objData); 
     form.appendChild(hiddenField); 
     // End Repeat for all data fields 


     document.body.appendChild(form); 



     // creating the 'formresult' window with custom features prior to submitting the form 
     //window.open(test.html, 'formresult', 'scrollbars=no,menubar=no,height=600,width=800,resizable=yes,toolbar=no,status=no'); 
     //JS_PopupCenterScreen(strPageURL, strWindowName, iWindowWidth, iWindowHeight); 
     window.open(strPageURL, strWindowName); 

     // document.forms[0].submit(); 
     //document.getElementById("xxx").click(); 
     form.submit(); 
    } // End Function IssuePostRequest 

与此服务器代码:

public FileResult GetPDF(string data) 
    { 
     //data = @""; 

     string base64Data = System.Text.RegularExpressions.Regex.Match(data, @"data:image/(?<type>.+?),(?<data>.+)").Groups["data"].Value; 
     byte[] binData = Convert.FromBase64String(base64Data); 

     byte[] ba = PdfHandler.ImageToPdf(binData); 
     //System.IO.File.WriteAllBytes(@"d:\temp\myba.pdf", ba); 

     //return System.Convert.ToBase64String(ba); 
     return File(ba, "application/pdf", "Chart.pdf"); 
    } 
+0

这是真的很有帮助,谢谢。唯一的问题是它打开一个新窗口。这可以在iFrame中打开吗? –

+2

@Menelaos Vergis:向您的页面添加一个​​。 –

+0

@Quandry:非常感谢,我不知道为什么这个答案没有得到应有的重视,它按预期工作。 –

0

我设法以此来解决这个问题:

service.js

downloadExcel : function() { 
    var mapForm = document.createElement("form"); 
    mapForm.target ="_self"||"_blank"; 
    mapForm.id="stmtForm"; 
    mapForm.method = "POST"; 
    mapForm.action = "your_Controller_URL"; 

    var mapInput = document.createElement("input"); 
    mapInput.type = "hidden"; 
    mapInput.name = "Data"; 
    mapForm.appendChild(mapInput); 
    document.body.appendChild(mapForm); 

    mapForm.submit(); 
} 

春控制器代码:

@Controller 

@PostMapping(value = "/your_Controller_URL") 
    public void doDownloadEmsTemplate(final HttpServletRequest request, final HttpServletResponse response) 
      throws IOException, URISyntaxException { 

     String filePath = "/location/zzzz.xls"; 
     logger.info("Excel Template File Location Path :" + filePath); 
     final int BUFFER_SIZE = 4096; 
     ServletContext context = request.getServletContext(); 
     String appPath = context.getRealPath(""); 
     String fullPath = appPath + filePath; 
     File downloadFile = new File(fullPath); 
     FileInputStream inputStream = new FileInputStream(downloadFile); 
     String mimeType = context.getMimeType(fullPath); 
     if (mimeType == null) { 
      //mimeType = "application/octet-stream"; 
      mimeType = "application/vnd.ms-excel"; 
     } 
     logger.info("MIME type: " + mimeType); 
     response.setContentType(mimeType); 
     response.setContentLength((int) downloadFile.length()); 
     String headerKey = "Content-Disposition"; 
     String headerValue = String.format("attachment; filename=\"%s\"", downloadFile.getName()); 
     logger.info("File Download Successfully : "); 
     response.setHeader(headerKey, headerValue); 
     OutputStream outStream = response.getOutputStream(); 
     byte[] buffer = new byte[BUFFER_SIZE]; 
     int bytesRead = -1; 
     while ((bytesRead = inputStream.read(buffer)) != -1) { 
      outStream.write(buffer, 0, bytesRead); 
     } 
     inputStream.close(); 
     outStream.close(); 
    } 
相关问题