1 文件上传的浸染

例如网络硬盘!
便是用来上传下载文件的。

在智联招聘上填写一个完全的简历还须要上传照片呢。

jsp下载mp3文件Java文件上传下载实训 Python

2 文件上传对页面的哀求

上传文件的哀求比较多,须要记一下:

1. 必须利用表单,而不能是超链接;

2. 表单的method必须是POST,而不能是GET;

3. 表单的enctype必须是multipart/form-data;

4. 在表单中添加file表单字段,即<input type=”file”…/>

<form action=\公众${pageContext.request.contextPath }/FileUploadServlet\公众 method=\公众post\"大众 enctype=\"大众multipart/form-data\"大众>

用户名:<input type=\公众text\"大众 name=\公众username\"大众/><br/>

文件1:<inputtype=\"大众file\"大众name=\"大众file1\"大众/><br/>

文件2:<inputtype=\"大众file\"大众name=\公众file2\公众/><br/>

<input type=\"大众submit\"大众 value=\"大众提交\公众/>

</form>

3 比对文件上传表单和普通文本表单的差异

通过httpWatch查看“文件上传表单”和“普通文本表单”的差异。

文件上传表单的enctype=”multipart/form-data”,表示多部件表单数据;

普通文本表单可以不设置enctype属性:

当method=”post”时,enctype的默认值为application/x-www-form-urlencoded,表示利用url编码正文;

当method=”get”时,enctype的默认值为null,没有正文,以是就不须要enctype了。

对普通文本表单的测试:

<form action=\"大众${pageContext.request.contextPath }/FileUploadServlet\"大众 method=\"大众post\公众>

用户名:<input type=\"大众text\"大众 name=\公众username\"大众/><br/>

文件1:<input type=\公众file\"大众 name=\"大众file1\"大众/><br/>

文件2:<input type=\"大众file\"大众 name=\"大众file2\公众/><br/>

<input type=\公众submit\"大众 value=\"大众提交\公众/>

</form>

通过httpWatch测试,查看表单的要求数据正文,我们创造要求中只有文件名称,而没有文件内容。
也便是说,当表单的enctype不是multipart/form-data时,要求中不包含文件内容,而只有文件的名称,这解释普通文本表单中input:file与input:text没什么差异了。

对文件上传表单的测试:

<form action=\"大众${pageContext.request.contextPath }/FileUploadServlet\"大众 method=\"大众post\"大众 enctype=\"大众multipart/form-data\"大众>

用户名:<input type=\"大众text\"大众 name=\"大众username\"大众/><br/>

文件1:<input type=\公众file\"大众 name=\"大众file1\公众/><br/>

文件2:<input type=\"大众file\"大众 name=\"大众file2\"大众/><br/>

<input type=\"大众submit\"大众 value=\"大众提交\公众/>

</form>

通过httpWatch测试,查看表单的要求数据正文部分,创造正文部分是由多个部件组成,每个部件对应一个表单字段,每个部件都有自己的头信息。
头信息下面是空行,空行下面是字段的正文部分。
多个部件之间利用随机天生的分隔线隔开。

文本字段的头信息中只包含一条头信息,即Content-Disposition,这个头信息的值有两个部分,第一部分是固定的,即form-data,第二部分为字段的名称。
在空行后面便是正文部分了,正文部分便是在文本框中填写的内容。

文件字段的头信息中包含两条头信息,Content-Disposition和Content-Type。
Content-Disposition中多出一个filename,它指定的是上传的文件名称。
而Content-Type指定的是上传文件的类型。
文件字段的正文部分便是文件的内容。

请把稳,由于我们上传的文件都是普通文本文件,即txt文件,以是在httpWatch中是可以正常显示的,如果上传的是exe、mp3等文件,那么在httpWatch看到的便是乱码了。

4 文件上传对Servlet的哀求

当提交的表单是文件上传表单时,那么对Servlet也是有哀求的。

首先我们要肯定一点,文件上传表单的数据也是被封装到request工具中的。

request.getParameter(String)方法获取指定的表单字段字符内容,但文件上传表单已经不在是字符内容,而是字节内容,以是失落效。

这时可以利用request的getInputStream()方法获取ServletInputStream工具,它是InputStream的子类,这个ServletInputStream工具对应全体表单的正文部分(从第一个分隔线开始,到末了),这解释我们须要的解析流中的数据。
当然解析它是很麻烦的一件事情,而Apache已经帮我们供应理解析它的工具:commons-fileupload。

可以考试测验把request.getInputStream()这个流中的内容打印出来,再比拟httpWatch中的要求数据。

publicvoid doPost(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

InputStream in = request.getInputStream();

String s = IOUtils.toString(in);

System.out.println(s);

}

-----------------------------7ddd3370ab2

Content-Disposition: form-data; name=\"大众username\"大众

hello

-----------------------------7ddd3370ab2

Content-Disposition: form-data; name=\"大众file1\"大众; filename=\"大众a.txt\公众

Content-Type: text/plain

aaa

-----------------------------7ddd3370ab2

Content-Disposition: form-data; name=\"大众file2\"大众; filename=\"大众b.txt\公众

Content-Type: text/plain

bbb

-----------------------------7ddd3370ab2--

commons-fileupload

为什么利用fileupload:

上传文件的哀求比较多,须要记一下:

必须是POST表单;

表单的enctype必须是multipart/form-data;

在表单中添加file表单字段,即<input type=”file”…/>

Servlet的哀求:

不能再利用request.getParameter()来获取表单数据;

可以利用request.getInputStream()得到所有的表单数据,而不是一个表单项的数据;

这解释不该用fileupload,我们须要自己来对request.getInputStream()的内容进行解析!


1 fileupload概述

fileupload是由apache的commons组件供应的上传组件。
它最紧张的事情便是帮我们解析request.getInputStream()。

fileupload组件须要的JAR包有:

commons-fileupload.jar,核心包;

commons-io.jar,依赖包。

2 fileupload大略运用

fileupload的核心类有:DiskFileItemFactory、ServletFileUpload、FileItem。

利用fileupload组件的步骤如下:

1. 创建工厂类DiskFileItemFactory工具:DiskFileItemFactory factory = new DiskFileItemFactory()

2. 利用工厂创建解析器工具:ServletFileUpload fileUpload = new ServletFileUpload(factory)

3. 利用解析器来解析request工具:List<FileItem> list = fileUpload.parseRequest(request)

隆重先容FileItem类,它才是我们终极要的结果。
一个FileItem工具对应一个表单项(表单字段)。
一个表单中存在文件字段和普通字段,可以利用FileItem类的isFormField()方法来判断表单字段是否为普通字段,如果不是普通字段,那么便是文件字段了。

String getName():获取文件字段的文件名称;

String getString():获取字段的内容,如果是文件字段,那么获取的是文件内容,当然上传的文件必须是文本文件;

String getFieldName():获取字段名称,例如:<input type=”text” name=”username”/>,返回的是username;

String getContentType():获取上传的文件的类型,例如:text/plain。

int getSize():获取上传文件的大小;

boolean isFormField():判断当前表单字段是否为普通文本字段,如果返回false,解释是文件字段;

InputStream getInputStream():获取上传文件对应的输入流;

void write(File):把上传的文件保存到指定文件中。

3 大略上传示例

写一个大略的上传示例:

表单包含一个用户名字段,以及一个文件字段;

Servlet保存上传的文件到uploads目录,显示用户名,文件名,文件大小,文件类型。

第一步:

完成index.jsp,只须要一个表单。
把稳表单必须是post的,而且enctype必须是mulitpart/form-data的。

<form action=\"大众${pageContext.request.contextPath }/FileUploadServlet\"大众 method=\"大众post\公众 enctype=\"大众multipart/form-data\"大众>

用户名:<input type=\公众text\"大众 name=\公众username\公众/><br/>

文件1:<input type=\"大众file\"大众 name=\"大众file1\"大众/><br/>

<input type=\"大众submit\"大众 value=\公众提交\公众/>

</form>

第二步:

完成FileUploadServlet

publicvoid doPost(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

// 由于要利用response打印,以是设置其编码

response.setContentType(\公众text/html;charset=utf-8\"大众);

// 创建工厂

DiskFileItemFactory dfif = new DiskFileItemFactory();

// 利用工厂创建解析器工具

ServletFileUpload fileUpload = new ServletFileUpload(dfif);

try {

// 利用解析器工具解析request,得到FileItem列表

List<FileItem> list = fileUpload.parseRequest(request);

// 遍历所有表单项

for(FileItem fileItem : list) {

// 如果当前表单项为普通表单项

if(fileItem.isFormField()) {

// 获取当前表单项的字段名称

String fieldName = fileItem.getFieldName();

// 如果当前表单项的字段名为username

if(fieldName.equals(\"大众username\公众)) {

// 打印当前表单项的内容,即用户在username表单项中输入的内容

response.getWriter().print(\"大众用户名:\公众 + fileItem.getString() + \"大众<br/>\"大众);

}

} else {//如果当前表单项不是普通表单项,解释便是文件字段

String name = fileItem.getName();//获取上传文件的名称

// 如果上传的文件名称为空,即没有指定上传文件

if(name == null || name.isEmpty()) {

continue;

}

// 获取真实路径,对应${项目目录}/uploads,当然,这个目录必须存在

String savepath = this.getServletContext().getRealPath(\公众/uploads\公众);

// 通过uploads目录和文件名称来创建File工具

File file = new File(savepath, name);

// 把上传文件保存到指定位置

fileItem.write(file);

// 打印上传文件的名称

response.getWriter().print(\"大众上传文件名:\"大众 + name + \公众<br/>\"大众);

// 打印上传文件的大小

response.getWriter().print(\"大众上传文件大小:\"大众 + fileItem.getSize() + \"大众<br/>\"大众);

// 打印上传文件的类型

response.getWriter().print(\"大众上传文件类型:\"大众 + fileItem.getContentType() + \公众<br/>\公众);

}

}

} catch (Exception e) {

thrownew ServletException(e);

}

}

文件上传之细节

1 把上传的文件放到WEB-INF目录下

如果没有把用户上传的文件存放到WEB-INF目录下,那么用户就可以通过浏览器直接访问上传的文件,这是非常危险的。

如果说用户上传了一个a.jsp文件,然后用户在通过浏览器去访问这个a.jsp文件,那么就会实行a.jsp中的内容,如果在a.jsp中有如下语句:Runtime.getRuntime().exec(“shutdown –s –t 1”);,那么你就会…

常日我们会在WEB-INF目录下创建一个uploads目录来存放上传的文件,而在Servlet中找到这个目录须要利用ServletContext的getRealPath(String)方法,例如在我的upload1项目中有如下语句:

ServletContext servletContext = this.getServletContext();

String savepath = servletContext.getRealPath(“/WEB-INF/uploads”);

个中savepath为:F:\tomcat6_1\webapps\upload1\WEB-INF\uploads。

2 文件名称(完全路径、文件名称)

上传文件名称可能是完全路径:

IE6获取的上传文件名称是完全路径,而其他浏览器获取的上传文件名称只是文件名称而已。
浏览器差异的问题我们还是须要处理一下的。

String name = file1FileItem.getName();

response.getWriter().print(name);

利用不同浏览器测试,个中IE6就会返回上传文件的完全路径,不知道IE6在搞什么,这给我们带来了很大的麻烦,便是须要处理这一问题。

处理这一问题也很大略,无论是否为完全路径,我们都去截取末了一个“\\”后面的内容就可以了。

String name = file1FileItem.getName();

int lastIndex = name.lastIndexOf(\公众\\\"大众);//获取末了一个“\”的位置

if(lastIndex != -1) {//把稳,如果不是完全路径,那么就不会有“\”的存在。

name = name.substring(lastIndex + 1);//获取文件名称

}

response.getWriter().print(name);

3 中文乱码问题

上传文件名称中包含中文:

当上传的谁的名称中包含中文时,须要设置编码,commons-fileupload组件为我们供应了两种设置编码的办法:

l request.setCharacterEncoding(String):这种办法是我们最为熟习的办法了;

l fileUpload.setHeaderEncdoing(String):这种办法的优先级高与前一种。

上传文件的文件内容包含中文:

常日我们不需关心上传文件的内容,由于我们会把上传文件保存到硬盘上!
也便是说,文件原来是什么样子,到做事器这边还是什么样子!

但是如果你有这样的需求,非要在掌握台显示上传的文件内容,那么你可以利用fileItem.getString(“utf-8”)来处理编码。

文本文件内容和普通表单项内容利用FileItem类的getString(“utf-8”)来处理编码。

4 上传文件同名问题(文件重命名)

常日我们会把用户上传的文件保存到uploads目录下,但如果用户上传了同名文件呢?这会涌现覆盖的征象。
处理这一问题的手段是利用UUID天生唯一名称,然后再利用“_”连接文件上传的原始名称。

例如用户上传的文件是“我的一寸照片.jpg”,在通过处理后,文件名称为:“891b3881395f4175b969256a3f7b6e10_我的一寸照片.jpg”,这种手段不会使文件丢失扩展名,并且由于UUID的唯一性,上传的文件同名,但在做事器端是不会涌现同名问题的。

publicvoid doPost(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

request.setCharacterEncoding(\公众utf-8\公众);

DiskFileItemFactory dfif = new DiskFileItemFactory();

ServletFileUpload fileUpload = new ServletFileUpload(dfif);

try {

List<FileItem> list = fileUpload.parseRequest(request);

//获取第二个表单项,由于第一个表单项是username,第二个才是file表单项

FileItem fileItem = list.get(1);

String name = fileItem.getName();//获取文件名称

// 如果客户端利用的是IE6,那么须要从完全路径中获取文件名称

int lastIndex = name.lastIndexOf(\"大众\\\"大众);

if(lastIndex != -1) {

name = name.substring(lastIndex + 1);

}

// 获取上传文件的保存目录

String savepath = this.getServletContext().getRealPath(\公众/WEB-INF/uploads\"大众);

String uuid = CommonUtils.uuid();//天生uuid

String filename = uuid + \"大众_\"大众 + name;//新的文件名称为uuid + 下划线 + 原始名称

//创建file工具,下面会把上传文件保存到这个file指定的路径

//savepath,即上传文件的保存目录

//filename,文件名称

File file = new File(savepath, filename);

// 保存文件

fileItem.write(file);

} catch (Exception e) {

thrownew ServletException(e);

}

}

5 一个目录不能存放过多的文件(存放目录打散)

一个目录下不应该存放过多的文件,一样平常一个目录存放1000个文件便是上限了,如果在多,那么打开目录时就会很“卡”。
你可以考试测验打印C:\WINDOWS\system32目录,你会觉得到的。

也便是说,我们须要把上传的文件放到不同的目录中。
但是也不能为每个上传的文件一个目录,这种办法会导致目录过多。
以是我们该当采取某种算法来“打散”!

打散的方法有很多,例如利用日期来打散,每天天生一个目录。
也可以利用文件名的首字母来天生目录,相同首字母的文件放到同一目录下。

日期打散算法:如果某一天上传的文件过多,那么也会涌现一个目录文件过多的情形;

首字母打散算法:如果文件名是中文的,由于中文过多,以是会导致目录过多的征象。

我们这里利用hash算法来打散:

1. 获取文件名称的hashCode:int hCode = name.hashCode();;

2. 获取hCode的低4位,然后转换成16进制字符;

3. 获取hCode的5~8位,然后转换成16进制字符;

4. 利用这两个16进制的字符天生目录链。
例如低4位字符为“5”

这种算法的好处是,在uploads目录下最多天生16个目录,而每个目录下最多再天生16个目录,即256个目录,所有上传的文件都放到这256个目录下。
如果每个目录上限为1000个文件,那么一共可以保存256000个文件。

例如上传文件名称为:新建 文本文档.txt,那么把“新建 文本文档.txt”的哈希码获取到,再获取哈希码的低4位,和5~8位。
如果低4位为:9,5~8位为1,那么文件的保存路径为uploads/9/1/。

int hCode = name.hashCode();//获取文件名的hashCode

//获取hCode的低4位,并转换成16进制字符串

String dir1 = Integer.toHexString(hCode & 0xF);

//获取hCode的低5~8位,并转换成16进制字符串

String dir2 = Integer.toHexString(hCode >>> 4 & 0xF);

//与文件保存目录连接成完全路径

savepath = savepath + \"大众/\"大众 + dir1 + \"大众/\"大众 + dir2;

//由于这个路径可能不存在,以是创建成File工具,再创建目录链,确保目录在保存文件之前已经存在

new File(savepath).mkdirs();

6 上传的单个文件的大小限定

限定上传文件的大小很大略,ServletFileUpload类的setFileSizeMax(long)就可以了。
参数便是上传文件的上限字节数,例如servletFileUpload.setFileSizeMax(102410)表示上限为10KB。

一旦上传的文件超出了上限,那么就会抛出FileUploadBase.FileSizeLimitExceededException非常。
我们可以在Servlet中获取这个非常,然后向页面输出“上传的文件超出限定”。

publicvoid doPost(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

request.setCharacterEncoding(\"大众utf-8\"大众);

DiskFileItemFactory dfif = new DiskFileItemFactory();

ServletFileUpload fileUpload = new ServletFileUpload(dfif);

// 设置上传的单个文件的上限为10KB

fileUpload.setFileSizeMax(1024 10);

try {

List<FileItem> list = fileUpload.parseRequest(request);

//获取第二个表单项,由于第一个表单项是username,第二个才是file表单项

FileItem fileItem = list.get(1);

String name = fileItem.getName();//获取文件名称

// 如果客户端利用的是IE6,那么须要从完全路径中获取文件名称

int lastIndex = name.lastIndexOf(\"大众\\\公众);

if(lastIndex != -1) {

name = name.substring(lastIndex + 1);

}

// 获取上传文件的保存目录

String savepath = this.getServletContext().getRealPath(\"大众/WEB-INF/uploads\"大众);

String uuid = CommonUtils.uuid();//天生uuid

String filename = uuid + \公众_\"大众 + name;//新的文件名称为uuid + 下划线 + 原始名称

int hCode = name.hashCode();//获取文件名的hashCode

//获取hCode的低4位,并转换成16进制字符串

String dir1 = Integer.toHexString(hCode & 0xF);

//获取hCode的低5~8位,并转换成16进制字符串

String dir2 = Integer.toHexString(hCode >>> 4 & 0xF);

//与文件保存目录连接成完全路径

savepath = savepath + \公众/\"大众 + dir1 + \公众/\"大众 + dir2;

//由于这个路径可能不存在,以是创建成File工具,再创建目录链,确保目录在保存文件之前已经存在

new File(savepath).mkdirs();

//创建file工具,下面会把上传文件保存到这个file指定的路径

//savepath,即上传文件的保存目录

//filename,文件名称

File file = new File(savepath, filename);

// 保存文件

fileItem.write(file);

} catch (Exception e) {

// 判断抛出的非常的类型是否为FileUploadBase.FileSizeLimitExceededException

// 如果是,解释上传文件时超出了限定。

if(e instanceof FileUploadBase.FileSizeLimitExceededException) {

// 在request中保存缺点信息

request.setAttribute(\"大众msg\"大众, \"大众上传失落败!
上传的文件超出了10KB!
\"大众);

// 转发到index.jsp页面中!
在index.jsp页面中须要利用${msg}来显示缺点信息

request.getRequestDispatcher(\公众/index.jsp\公众).forward(request, response);

return;

}

thrownew ServletException(e);

}

}

7 上传文件的总大小限定

上传文件的表单中可能许可上传多个文件,例如:

有时我们须要限定一个要求的大小。
也便是说这个要求的最大字节数(所有表单项之和)!
实现这一功能也很大略,只须要调用ServletFileUpload类的setSizeMax(long)方法即可。

例如fileUpload.setSizeMax(1024 10);,显示全体要求的上限为10KB。
当要求大小超出10KB时,ServletFileUpload类的parseRequest()方法会抛出FileUploadBase.SizeLimitExceededException非常。

8 缓存大小与临时目录

大家想一想,如果我上传一个蓝光电影,先把电影保存到内存中,然后再通过内存copy到做事器硬盘上,那么你的内存能吃的消么?

以是fileupload组件不可能把文件都保存在内存中,fileupload会判断文件大小是否超出10KB,如果是那么就把文件保存到硬盘上,如果没有超出,那么就保存在内存中。

10KB是fileupload默认的值,我们可以来设置它。

当文件保存到硬盘时,fileupload是把文件保存到系统临时目录,当然你也可以去设置临时目录。

publicvoid doPost(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

request.setCharacterEncoding(\公众utf-8\公众);

DiskFileItemFactory dfif = new DiskFileItemFactory(102420, new File(\"大众F:\\temp\"大众));

ServletFileUpload fileUpload = new ServletFileUpload(dfif);

try {

List<FileItem> list = fileUpload.parseRequest(request);

FileItem fileItem = list.get(1);

String name = fileItem.getName();

String savepath = this.getServletContext().getRealPath(\"大众/WEB-INF/uploads\"大众);

// 保存文件

fileItem.write(path(savepath, name));

} catch (Exception e) {

thrownew ServletException(e);

}

}

private File path(String savepath, String filename) {

// 从完全路径中获取文件名称

int lastIndex = filename.lastIndexOf(\公众\\\"大众);

if(lastIndex != -1) {

filename = filename.substring(lastIndex + 1);

}

// 通过文件名称天生一级、二级目录

int hCode = filename.hashCode();

String dir1 = Integer.toHexString(hCode & 0xF);

String dir2 = Integer.toHexString(hCode >>> 4 & 0xF);

savepath = savepath + \"大众/\"大众 + dir1 + \公众/\"大众 + dir2;

// 创建目录

new File(savepath).mkdirs();

// 给文件名称添加uuid前缀

String uuid = CommonUtils.uuid();

filename = uuid + \公众_\公众 + filename;

// 创建文件完成路径

returnnew File(savepath, filename);

}

文件下载

2 通过Servlet下载1

被下载的资源必须放到WEB-INF目录下(只要用户不能通过浏览器直接访问就OK),然后通过Servlet完成下载。

在jsp页面中给出超链接,链接到DownloadServlet,并供应要下载的文件名称。
然后DownloadServlet获取文件的真实路径,然后把文件写入到response.getOutputStream()流中。

download.jsp

<body>

This is my JSP page. <br>

<a href=\"大众<c:url value='/DownloadServlet?path=a.avi'/>\"大众>a.avi</a><br/>

<a href=\"大众<c:url value='/DownloadServlet?path=a.jpg'/>\"大众>a.jpg</a><br/>

<a href=\"大众<c:url value='/DownloadServlet?path=a.txt'/>\"大众>a.txt</a><br/>

</body>

DownloadServlet.java

publicvoid doGet(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

String filename = request.getParameter(\公众path\"大众);

String filepath = this.getServletContext().getRealPath(\"大众/WEB-INF/uploads/\"大众 + filename);

File file = new File(filepath);

if(!file.exists()) {

response.getWriter().print(\"大众您要下载的文件不存在!
\公众);

return;

}

IOUtils.copy(new FileInputStream(file), response.getOutputStream());

}

上面代码有如下问题:

可以下载a.avi,但不才载框中的文件名称是DownloadServlet;

不能下载a.jpg和a.txt,而是在页面中显示它们。

3 通过Servlet下载2

下面来处理上一例中的问题,让下载框中可以显示精确的文件名称,以及可以下载a.jpg和a.txt文件。

通过添加content-disposition头来处理上面问题。
当设置了content-disposition头后,浏览器就会弹出下载框。

而且还可以通过content-disposition头来指定下载文件的名称!

String filename = request.getParameter(\"大众path\"大众);

String filepath = this.getServletContext().getRealPath(\公众/WEB-INF/uploads/\"大众 + filename);

File file = new File(filepath);

if(!file.exists()) {

response.getWriter().print(\公众您要下载的文件不存在!
\公众);

return;

}

response.addHeader(\"大众content-disposition\"大众, \公众attachment;filename=\公众 + filename);

IOUtils.copy(new FileInputStream(file), response.getOutputStream());

虽然上面的代码已经可以处理txt和jpg等文件的下载问题,并且也处理了不才载框中显示文件名称的问题,但是如果下载的文件名称是中文的,那么还是弗成的。

3 通过Servlet下载3

下面是处理不才载框中显示中文的问题!

实在这一问题很大略,只须要通过URL来编码中文即可!

download.jsp

<a href=\"大众<c:url value='/DownloadServlet?path=这个杀手不太冷.avi'/>\公众>这个杀手不太冷.avi</a><br/>

<a href=\"大众<c:url value='/DownloadServlet?path=白冰.jpg'/>\"大众>白冰.jpg</a><br/>

<a href=\公众<c:url value='/DownloadServlet?path=解释文档.txt'/>\"大众>解释文档.txt</a><br/>

DownloadServlet.java

String filename = request.getParameter(\公众path\"大众);

// GET要求中,参数中包含中文须要自己动手来转换。

// 当然如果你利用了“全局编码过滤器”,那么这里就不用处理了

filename = new String(filename.getBytes(\公众ISO-8859-1\公众), \"大众UTF-8\"大众);

String filepath = this.getServletContext().getRealPath(\公众/WEB-INF/uploads/\"大众 + filename);

File file = new File(filepath);

if(!file.exists()) {

response.getWriter().print(\"大众您要下载的文件不存在!
\"大众);

return;

}

// 所有浏览器都会利用本地编码,即中文操作系统利用GBK

// 浏览器收到这个文件名后,会利用iso-8859-1来解码

filename= new String(filename.getBytes(\"大众GBK\"大众), \"大众ISO-8859-1\"大众);

response.addHeader(\公众content-disposition\"大众, \公众attachment;filename=\公众 + filename);

IOUtils.copy(new FileInputStream(file), response.getOutputStream());