jsp下载文件的实现方法

(一) 
最简单的方式是在网页上做超级链接,如:<a href="music/1.zip">点击下载</a>。 
但是这样服务器上的目录资源会直接暴露给最终用户,会给网站带来一些不安全的因素。

因此可以采用其它方式实现下载,可以采用: 
1、RequestDispatcher的方式进行; 
2、采用文件流输出的方式下载。 (推荐)

1、采用RequestDispatcher的方式进行 

<%
    response.setContentType("application/x-download");//设置为下载application/x-download
    String filedownload = "/要下载的文件名";//即将下载的文件的相对路径
    String filedisplay = "最终要显示给用户的保存文件名";//下载文件时显示的文件保存名称
    String filenamedisplay = URLEncoder.encode(filedisplay,"UTF-8");
    response.addHeader("Content-Disposition","attachment;filename=" + filedisplay);  

try{
    RequestDispatcher dis = application.getRequestDispatcher(filedownload);
    if(dis!= null)  {
    dis.forward(request,response);
    }
    response.flushBuffer();
}  catch(Exception e)  {
    e.printStackTrace();
}  finally  {
     }
%>  

2、采用文件流输出的方式下载

<%@page language="java" contentType="application/x-msdownload" pageEncoding="gb2312"%>
<%
  //关于文件下载时采用文件流输出的方式处理:
  //加上response.reset(),并且所有的%>后面不要换行,包括最后一个;  

  response.reset();//可以加也可以不加
  response.setContentType("application/x-download");  

//application.getRealPath("/main/mvplayer/CapSetup.msi");获取的物理路径  

String filedownload = "想办法找到要提供下载的文件的物理路径+文件名";
String filedisplay = "给用户提供的下载文件名";
  String filedisplay = URLEncoder.encode(filedisplay,"UTF-8");
  response.addHeader("Content-Disposition","attachment;filename=" + filedisplay);  

  java.io.OutputStream outp = null;
  java.io.FileInputStream in = null;
  try
  {
  outp = response.getOutputStream();
  in = new FileInputStream(filenamedownload);  

  byte[] b = new byte[1024];
  int i = 0;  

  while((i = in.read(b)) > 0)
  {
  outp.write(b, 0, i);
  }
//
outp.flush();
//要加以下两句话,否则会报错
//java.lang.IllegalStateException: getOutputStream() has already been called for //this response
out.clear();
out = pageContext.pushBody();
}
  catch(Exception e)
  {
  System.out.println("Error!");
  e.printStackTrace();
  }
  finally
  {
  if(in != null)
  {
  in.close();
  in = null;
  }
//这里不能关闭
//if(outp != null)
  //{
  //outp.close();
  //outp = null;
  //}
  }
%>  

对于第二种方法,我认为应该是比较常用的。不过有几个地方是值得我们注意的:

一、采用第二种方法的主要优点是实际文件的存放路径对客户端来说是透明的。 
这个文件可以存在于任何你的服务器能够取得到的地方,而客户端不一定能直接得到。例如文件来自于数据库或者内部网络的一个FTP服务器。换句话说,这种方式可以实现隐藏实际文件的URL地址。

二、为了防止客户端浏览器直接打开目标文件(例如在装了MS Office套件的Windows中的IE浏览器可能就会直接在IE浏览器中打开你想下载的doc或者xls文件),你必须在响应头里加入强制下载的MIME类型: 
response.setContentType("application/force-download");//设置为下载application/force-download 
这样,就可以保证在用户点击下载链接的时候浏览器一定会弹出提示窗口来询问你是下载还是直接打开并允许你选择要打开的应用程序,除非你设置了浏览器的一些默认行为。 
或者,你想让客户端自行处理各种不同的文件类型,你可以在服务器的配置文件中配置MIME类型映射,通过简单的判断文件后缀名来处理。例如,在Tomcat中设置MIME响应类型: 
如果文件在客户端中的响应程序类型和期望不一致,修改$TOMCAT_HOME\conf\web.xml文件中的如下部分 : 
<mime-mapping> 
  <extension>zip</extension> 
  <mime-type>application/zip</mime-type> 
</mime-mapping> 
<mime-mapping> 
  <extension>mht</extension> 
  <mime-type>message/rfc822</mime-type> 
</mime-mapping> 
……

三、在响应头中尽量不要设置浏览器缓存期限。 
有时候用户在点击了下载链接后,在弹出窗口中,用户想直接点击“打开”,而不想保存到指定路径。这时候如果我们在响应头中限制了不允许使用浏览器缓存(即总是刷新),在IE浏览器中我们将无法直接打开该文件。因为限制了不允许使用缓存,浏览器无法将文件保存到临时文件夹(即缓存)。 
也就是说,在响应头中不要进行如下的设置(已注释): 
  //response.addHeader("pragma","NO-cache"); 
  //response.addHeader("Cache-Control","no-cache"); 
  //response.addDateHeader("Expries",0);

四、文件名为中文或其他unicode字符时的处理。 
有时候提供下载的文件名中包含中文字符或者其他unicode字符,会导致浏览器无法正确的采用默认的文件名保存文件。我们应该记住在响应头中包含filename字段并采用ISO8859-1编码(推荐)或者采用UTF-8编码: 
response.setHeader("Content-disposition","attachment; filename="+new String(filename.getBytes("UTF-8"),"iso8859-1")); //采用ISO8859-1编码 
response.setHeader("Content-disposition","attachment; filename="+URLEncoder.encode(filename, "UTF-8")); //采用UTF-8编码 
但是,这种方式在不同的浏览器中表现也有所不同。例如在IE和Firefox中,采用ISO8859-1编码可以正确显示文件名,而在Opera中不管采用那种编码,默认保存的文件名都无法做到正确显示。 
所以最好的方法其实就是尽量在文件名中使用ascii编码。

五、由于采用流的方式进行输入输出,我们必须保证在使用完毕后关闭流的资源。 
一般我们把关闭流的操作放在finally块中,以保证在程序段结束前一定会关闭流的资源:

时间: 11-16

jsp下载文件的实现方法的相关文章

jsp下载文件的实现方法及注意事项 (转)

jsp中实现文件下载,最简单的方式是在网页上做超级链接,如:<a href="music/abc.mp3">点击下载</a>. 但是,这样服务器上的目录资源会直接暴露给最终用户,会给网站带来一些不安全的因素. 因此,可以采用其它方式实现下载,常使用的有以下两种:       1.RequestDispatcher的方式进行:       2.采用文件流输出的方式下载(推荐). 1.采用RequestDispatcher的方式: 1 <% 2 respons

下载文件(简洁方法)

public static void DownLoadFile(HttpRequest _Request, HttpResponse _Response, string fileName, string fullPath) { FileInfo fileInfo = new FileInfo(fullPath); if (fileInfo.Exists) { _Response.Clear(); _Response.ClearHeaders(); _Response.Buffer = false

asp.net C#实现下载文件的六种方法实例

protected void Button1_Click(object sender, EventArgs e)  {  /*  微软为Response对象提供了一个新的方法TransmitFile来解决使用Response.BinaryWrite  下载超过400mb的文件时导致Aspnet_wp.exe进程回收而无法成功下载的问题.  代码如下:  */ Response.ContentType = "application/x-zip-compressed";  Response

Win10系统IE浏览器能打开网页但无法下载文件的处理方法

在Win10系统中,很多没有安装下载软件的朋友默认就是使用IE来下载文件的,不过部分用户发现使用IE下载时出现点击无反应,无法下载的情况,那么如何解决此类问题呢?原因分析:出现此类情况大多是由于默认下载文件夹不存在引起的具体有如下几种方面.1.Win10系统安装好后修改了系统盘符修改系统盘符,IE浏览器默认的安装位置出现错误,导致IE无法运行,因此无法下载.建议:检查皇冠现金系统盘符是否和安装时的是一致的,若发生变化,把盘符重新改回和原来一致.2.默认下载文件夹不存在默认下载文件夹不存在,IE下

解决jsp下载文件,迅雷下载路径不显示文件名称的问题

如果浏览器安装了迅雷的插件,在jsp页面调用java后台实现文件下载功能时,会自动弹出迅雷下载,迅雷的下载路径会显示.do或者.xhtml之类的,为了解决这个问题,jsp页面修改如下: 写一个<a>标签在页面上: <a id="downloadUrl" target="_blank"></a> 导出按钮如下: <a href="javascript:void(0)" id="biz_exp_bu

Microsoft Edge浏览器下载文件乱码修复方法

随着Windows10的普及,Microsoft Edge自带浏览器使用频率逐渐提升,在日常使用过程中我们会发现一个常规的问题是使用Edge进行日常文件下载的时候,N多情况下可能都是乱码,同样的下载链接在Chrome\Firefox\IE等等都是没有问题的,这就很尴尬了,今天简单介绍下处理方法,具体如下: 乱码问题: 处理方法: 1.热键Win + R打开运行栏,输入gpedit.msc打开组策略编辑器: 2.在本地组策略编辑器中定位 计算机配置--管理模板--Windows组件--Intern

正确设置Firefox下载文件文件名的方法

不同的浏览器需要特殊设置,主要是火狐比较特殊,火狐可能给文件名加上“%0d%0a"这样的编码字符(换行的意思).不得不佩服网上的高手,这也能解决. 1 [HttpGet] 2 public FileResult Download(string id) 3 { 4 var document = service.GetDocument(id); 5 var fullName = Path.Combine(Root, document.FullName); 6 string browser = Htt

向云服务器上传下载文件方法汇总(转)

转载于:https://yq.aliyun.com/articles/64700 摘要: 一.向Windows服务器上传下载文件方式 方法有很多种,此处介绍远程桌面的本地资源共享方法. 1.运行mstsc,连接远程桌面的时候,点"选项>>" 2."本地资源"-->详细信息. 3."磁盘驱动器"前面打钩. 一.向Windows服务器上传下载文件方式 方法有很多种,此处介绍远程桌面的本地资源共享方法. 1.运行mstsc,连接远程桌

经验分享] 向云服务器上传下载文件方法汇总

一.向Windows服务器上传下载文件方式 方法有很多种,此处介绍远程桌面的本地资源共享方法. 1.运行mstsc,连接远程桌面的时候,点"选项>>" <ignore_js_op> 2."本地资源"-->详细信息. <ignore_js_op> 3."磁盘驱动器"前面打钩. <ignore_js_op> 4.连接远程电脑后,打开"我的电脑",就能看到刚刚共享的硬盘驱动器.把