百度空间 | 百度首页 
 
查看文章
 
struts2多文件动态下载及中文解决方案
2009年06月16日 星期二 16:40

关于文件上传的例子,网上已经有很多,所以本文就不谈文件上传,重点是文件下载及文件名为中文的时候各个浏览器出现乱码的解决方案。

下面是jsp文件的代码:

  1. <html>
  2. <head>
  3.    <title>download</title>
  4. </head>
  5. <body>
  6.    <%
  7. //取得服务器"/download/file"目录的物理路径
  8. String path = request.getRealPath("/download/file");
  9. //取得"/download/file"目录的file对象
  10. File file = new File(path);
  11. //取得file目录下所有文件
  12. File[] files = file.listFiles();
  13.    for (int i = 0; i < files.length; i++) {
  14. String fname = files[i].getName();
  15. //对文件名进行url编码(UTF-8指明fname原来的编码,UTF-8一般由本地编码GBK代替)
  16.    fname = java.net.URLEncoder.encode(fname, "UTF-8");
  17. out.println("<a href=download.action?name=" + fname + ">"
  18.    + files[i].getName() + "</a><br>");
  19. }
  20.    %>
  21. </body>
  22. </html>

相应的DownloadAction.java的代码:

  1. package com.test.action;
  2. import java.io.InputStream;
  3. import java.io.UnsupportedEncodingException;
  4. import com.opensymphony.xwork2.ActionSupport;
  5. import org.apache.struts2.ServletActionContext;
  6. public class DownloadAction extends ActionSupport {
  7.    private static final long serialVersionUID = 6329383258366253255L;
  8.    private String fileName;
  9.    public void setFileName(){
  10.    //得到请求下载的文件名
  11.    String fname=ServletActionContext.getRequest().getParameter("name");
  12.    try {
  13.    /*
  14. * 对fname参数进行UTF-8解码,注意:实际进行UTF-8解码时会使用本地编码,本机为GBK。
  15. * 这里使用request.setCharacterEncoding解码无效.
  16. * 只有解码了getDownloadFile()方法才能在下载目录下正确找到请求的文件
  17. * */  
  18.    fname = new String(fname.getBytes("ISO-8859-1"), "UTF-8");
  19.    } catch (Exception e) {
  20.    e.printStackTrace();
  21. }  
  22. this.fileName=fname;  
  23. System.out.println(fileName);
  24. }
  25.    /*
  26. * @getFileName
  27. * 此方法对应的是struts.xml文件中的:
  28. * <param name="contentDisposition">attachment;filename="${fileName}"</param>
  29. * 这个属性设置的是下载工具下载文件时显示的文件名,
  30. * 要想正确的显示中文文件名,我们需要对fileName再次编码
  31. * 否则中文名文件将出现乱码,或无法下载的情况
  32. * */
  33. public String getFileName() throws UnsupportedEncodingException {
  34.   
  35. fileName=new String(fileName.getBytes(),"ISO-8859-1");
  36.   
  37. return fileName;
  38. }
  39.    /*
  40.    * @getDownloadFile
  41.    * 此方法对应的是struts.xml文件中的:
  42.    * <param name="inputName">downloadFile</param>
  43.    * 返回下载文件的流,可以参看struts2的源码
  44.    * */
  45. public InputStream getDownloadFile() {
  46.   
  47.    this.setFileName();
  48.    return ServletActionContext.getServletContext().getResourceAsStream("/download/file/" + fileName);
  49. }
  50.    @Override
  51. public String execute() throws Exception {
  52. return SUCCESS;
  53.    }
  54. }

struts.xml相应的Action配置:

  1. <action name="download" class="com.test.action.DownloadAction">
  2. <result name="success" type="stream">
  3. <param name="contentDisposition">attachment;filename="${fileName}"</param>
  4. <param name="inputName">downloadFile</param>
  5. </result>
  6.    </action>

web.xml:

  1. <filter>
  2. <filter-name>struts2</filter-name>
  3. <filter-class>org.apache.struts2.dispatcher.FilterDispatcher</filter-class>
  4. </filter>
  5. <filter-mapping>
  6. <filter-name>struts2</filter-name>
  7. <url-pattern>/*</url-pattern>
  8. </filter-mapping>
  9. <welcome-file-list>
  10. <welcome-file>download.jsp</welcome-file>
  11. </welcome-file-list>

www.xileju.biz

www.xileju.biz

www.xileju.biz

www.xileju.biz

      本文的源代码在IE8.0Bate1、firefox3.0RC1、Opera下均测试通过,能够正常下载及正确显示中文文件名。

      相对于文件上传来说,下载要相对简单的多,但是如何让浏览器下载窗口显示中文名却是一件头痛的事,我曾在execute方法里加上这样一段代码:

ServletActionContext.getResponse().setHeader("Cont ent-Disposition","attachment; filename="+ new String(this.getFileName().getBytes("GBK"),"ISO-8859-1"));

      这是由于比较固执的想法和观念没有改变造成的,总想着我要去设置setHeader,告诉浏览器这是一个下载文件并把文件名经过相应的编码给它,让它能正确显示。事实证明这样做会带来一件让人沮丧的事,就是不管你怎么设它,在firefox和Opera里面中文文件名它都是乱码,好不容易把它转成中文显示了,却发现文件的后缀名最后有一个英文的逗号,也就是说,用户在下载了相应的文件后要自己手动去改后缀名,才能正常使用这个文件,可想而知,这是不现实的。我曾想过去判断用户所使用的浏览器,如果不是IE的话,我就把最后那个逗号给截了。可谁知道要是换了别的应行环境它出现的是不是还是逗号,而且这个逗号是怎么来的,为什么是逗号,我一无所知,要是在别的环境下它出现的不是逗号,我是不是要去重新改代码,这更是一件不现实的事。还好最后我没这样做,要是这样做了,那真是傻得没药救了!

      事实上struts2中很多功能都为我们封装得很好了,只要我们给它属性注入相应的值,程序就能正常的跑起来。文件的下载也不例外,我们找到这样一个类StreamResult,文件的下载由这个类全部为我们完成。其中有一个非常重要的属性contentDisposition,这个属性决定了下载窗口中显示的文件名,不用我们像上面一样去手动设置response的setHeader,我们只要给这个属性注入正确的值,一切就OK。

      而这个值需要去struts.xml文件里相应的action里去配置,由于是动态决定下载的文件,这里使用了EL表达式,${fileName},而fileName对应的则是DownloadAction里的getFileName()方法,只要在这里给fileName相应的编码,struts2会自动寻找这个方法,并把相应的值注入到xml文件中,这样浏览器的下载窗口中就能正常显示我们想要的中文名。


类别:Struts2 | 添加到搜藏 | 浏览() | 评论 (5)
 
最近读者:
 
网友评论:
1
2009年07月01日 星期三 11:57 | 回复
写的很好,谢谢分享。
 
2
2009年07月01日 星期三 17:16 | 回复
要是要下载的文件中包含“要”字的话,比如文件名为“项目需求.gif",就不对了。等待解决.........
 
3
2009年07月02日 星期四 09:29 | 回复
对不起,是我自己没有在页面上对文件名进行utf-8编码,所以导致错误。我本以为设置了以下语句就可以了。
<%@ page language = "java" contentType = "text/html; charset=UTF-8" pageEncoding = "UTF-8" %>
 
4
2009年07月16日 星期四 18:02 | 回复
k
 
5
2009年10月23日 星期五 16:12 | 回复
这样还是有错误的,部分中文还是会有乱码的,我的实现和楼主的一样,在ie下不会有乱码(目前没发现),在firefox下,大部分中文文件名没有,但是有些有乱码,例如,图片.jpg 就会出现乱码。如果楼主看到,可以测试一下,如果没错的话,能否告知一声。
 
发表评论:
姓 名:
网址或邮箱: (选填)
内 容:
验证码: 请点击后输入四位验证码,字母不区分大小写
      

     

©2009 Baidu