Struts 2中实现文件下载(修正中文问题)

BlogJava上已经有一位做者阐述了文件上传的问题,地址是Struts 2中实现文件上传,所以我就再也不讨论那个话题了。我今天简单介绍一下Struts 2的文件下载问题。html

咱们的项目名为 struts2hello,所使用的开发环境是MyEclipse 6,固然其实用哪一个IDE都是同样的,只要把类库放进去就好了,文件下载不须要再加入任何额外的包。读者能够参考文档:http://beansoft.java-cn.org/myeclipse_doc_cn/struts2_demo.pdf,来了解怎么下载和配置基本的Struts 2开发环境。java


为了便于你们对比,我把完整的struts.xml的配置信息列出来:web

<?xmlversion="1.0"encoding="UTF-8" ?>数据库

<!DOCTYPEstrutsPUBLICapache

    "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"数组

    "http://struts.apache.org/dtds/struts-2.0.dtd">浏览器

<struts>安全

    <packagename="default"extends="struts-default" >app

        <!-- 在这里添加Action定义 -->eclipse

        <!-- 简单文件下载 -->

        <actionname="download"class="example.FileDownloadAction">

            <resultname="success"type="stream">

                <paramname="contentType">text/plain</param>

                <paramname="inputName">inputStream</param>

                <paramname="contentDisposition">attachment;filename="struts2中文.txt"</param>

                <paramname="bufferSize">4096</param>

            </result>

        </action>

       

        <!-- 文件下载,支持中文附件名 -->

        <actionname="download2"class="example.FileDownloadAction2">

            <!-- 初始文件名 -->

            <paramname="fileName">Struts中文附件.txt</param>

            <resultname="success"type="stream">

                <paramname="contentType">text/plain</param>

                <paramname="inputName">inputStream</param>

                <!-- 使用通过转码的文件名做为下载文件名,downloadFileName属性

                对应action类中的方法 getDownloadFileName() -->

                <paramname="contentDisposition">attachment;filename="${downloadFileName}"</param>

                <paramname="bufferSize">4096</param>

            </result>

        </action>

        

        <!-- 下载现有文件 -->

        <actionname="download3"class="example.FileDownloadAction3">

            <paramname="inputPath">/download/系统说明.doc</param>

            <!-- 初始文件名 -->

            <paramname="fileName">系统说明.doc</param>

            <resultname="success"type="stream">

                <paramname="contentType">application/octet-stream;charset=ISO8859-1</param>

                <paramname="inputName">inputStream</param>

                <!-- 使用通过转码的文件名做为下载文件名,downloadFileName属性

                对应action类中的方法 getDownloadFileName() -->

                <paramname="contentDisposition">attachment;filename="${downloadFileName}"</param>

                <paramname="bufferSize">4096</param>

            </result>

        </action>

       

    </package>

</struts>

Struts 2中对文件下载作了直接的支持,相比起本身辛辛苦苦的设置种种HTTP头来讲,如今实现文件下载无疑要简便的多。提及文件下载,最直接的方式恐怕是直接写一个超连接,让地址等于被下载的文件,例如:<a href=”file1.zip”>下载file1.zip</a>,以后用户在浏览器里面点击这个连接,就能够进行下载了。可是它有一些缺陷,例如若是地址是一个图片,那么浏览器会直接打开它,而不是显示保存文件的对话框。再好比若是文件名是中文的,它会显示一堆URL编码过的文件名例如%3457...。而假设你企图这样下载文件:http://localhost:8080/struts2hello/download/系统说明.docTomcat会告诉你一个文件找不到的404错误:HTTP Status 404 - /struts2hello/download/ϵͳ˵Ã÷.doc。虽然目前还没发现直接配置Struts 2来正确的下载中文名字的附件,不过好在做者对JSP中的文件下载比较了解,所以咱们另有办法解决这个问题。另一个最大的用途,就是动态的生成并下载文件了,例如动态的下载生成的EXCELPDF,验证码图片等等。本节内容就依次讨论简单的下载文件代码,下载中文附件,最后介绍如何下载已经存在的文件。

先说文件下载,编写一个普通的Action就能够了,只须要提供一个返回InputStream流的方法,该输入流表明了被下载文件的入口,这个方法用来给被下载的数据提供输入流,意思是从这个流读出来,再写到浏览器那边供下载。这个方法须要由开发人员本身来编写,只须要返回值为InputStream便可。在咱们的例子中方法的签名是:public InputStream getInputStream() throws Exception,固然它也能够是别的名字,例如getDownloadFile()。好了,如今咱们所写的这个进行文件下载的Actionexample.FileDownloadAction的源代码清单以下:

package example;
import java.io.ByteArrayInputStream;
import java.io.InputStream;
import com.opensymphony.xwork2.Action;
public class FileDownloadAction implements Action {
public InputStream getInputStream() throws Exception {
return new ByteArrayInputStream("Struts 2 下载示例".getBytes());
}
public String execute() throws Exception {
return SUCCESS;
}
}


 注意这里惟一特殊的方法就是getInputStream(),在这个方法里面咱们使用了一个数组输入流来从字符串转换成的数组做为数据的来源进行读取。也许方法体中使用这样的实现代码:

return new java.io.FileInputStream(“c:""test.txt”);//从系统磁盘文件读取数据

这样会更直观一些。

文件下载的第二步,乃是在struts.xml中对action进行配置,其代码清单以下所示:

<!-- 简单文件下载 -->

<action name="download" class="example.FileDownloadAction">
<result name="success" type="stream">
<param name="contentType">text/plain</param>
<param name="inputName">inputStream</param>
<param name="contentDisposition">attachment;filename="struts2.txt"</param>
<param name="bufferSize">4096</param>
</result>
</action>


 这个action特殊的地方在于result的类型是一个流(stream),配置stream类型的结果时,由于无需指定实际的显示的物理资源,因此无需指定location属性,只须要指定inputName属性,该属性指向被下载文件的来源,对应着Action类中的某个属性,类型为InputStream。下面则列出了和下载有关的一些参数列表:

参数

说明

contentType

内容类型,和互联网MIME标准中的规定类型一致,例如text/plain表明纯文本,text/xml表示XMLimage/gif表明GIF图片,image/jpeg表明JPG图片

inputName

下载文件的来源流,对应着action类中某个类型为Inputstream的属性名,例如取值为inputStream的属性须要编写getInputStream()方法

contentDisposition

文件下载的处理方式,包括内联(inline)和附件(attachment)两种方式,而附件方式会弹出文件保存对话框,不然浏览器会尝试直接显示文件。取值为:

attachment;filename="struts2.txt",表示文件下载的时候保存的名字应为struts2.txt。若是直接写filename="struts2.txt",那么默认状况是表明inline,浏览器会尝试自动打开它,等价于这样的写法:inline; filename="struts2.txt"

bufferSize

下载缓冲区的大小

。在这里面,contentType属性和contentDisposition分别对应着HTTP响应中的头Content-TypeContent-disposition头。好,咱们先来看看这个例子,发布运行项目后键入测试地址:http://localhost:8080/struts2hello/download.action,将会看到浏览器弹出一个文件保存对话框,如图12.12所示。

12.12 文件下载对话框(IE 7Firefox 3

若是此时使用某些工具来探测浏览器返回的HTTP头,将会看到下列内容:

HTTP/1.1 200 OK

Server: Apache-Coyote/1.1

Content-disposition: attachment;filename="struts2.txt"

Content-Type: text/plain

Transfer-Encoding: chunked

Date: Sun, 02 Mar 2008 02:58:25 GMT

。因此用来下载的action配置中,只有两个是和浏览器有关的:contentTypecontentDisposition。关于contentType的取值,若是是未知的文件类型,或者说出现了浏览器不能打开的文件,例如.bean文件,或者说这个action是用来作动态文件下载的,事先并不知道将来的文件类型是什么,那么咱们能够把它的值设置成为:application/octet-stream;charset=ISO8859-1,注意必定要加入charset,不然某些时候会致使下载的文件出错;有人说这时也能够设置成为application/x-download,根据笔者的实践,这个头也能正常工做,然而个别时候会出现浏览器没法识别的问题。而contentDisposition,若是其取值是filename="struts2.txt",或者是inline; filename="struts2.txt",运行后你能够看到浏览器直接显示了文件的内容:

Struts 2 下载示例,而再也不弹出对话框提示用户保存文件到硬盘上。因此读者若是想确保文件是被下载而不是被打开,务必使用格式attachment;filename="struts2.txt",不要丢了attachment;这个类型信息。

至此,关于文件下载的技术内容,已经告一段落。然而作中文系统,不可避免的要解决中文附件的下载问题。关于这个内容,也无权威的资料可查,咱们只能用实践中获得的解决方案来处理。也许有读者觉得将filename属性设置为filename=”struts2中文.txt”就能解决问题了,好,就来试试,把contentDisposition修改为:

<param name="contentDisposition">attachment;filename="struts2中文.txt"</param>

。再次键入地址进行测试,看看显示的结果,如图12.13所示。唉,真是彻底不给面子!IE压根就不能显示出来文件名,草草敷衍了download_action了事。Firefox稍好点,还出来了一个对话框,可是很显然,那个显示的struts2--txt绝对不是咱们日思夜想的struts2中文.txt。怎么办?解决方法是有,那就是用ISO8859-1编码来显示这个中文字符,能够阅读12.8参考资料一节中的JSP 文件下载的相对完整代码(解决中文问题和Weblogic报错)这篇文章,能够这样认为,全部的文件下载代码都是基于一样的纯Servlet的方式来进行的。若是是Java代码,咱们能够这样作:

12.13 IEFirefox下的中文文件下载对话框

String downFileName = new String(“struts2中文.txt”.getBytes(), "ISO8859-1");

而后把生成的结果字符串放到XML文件中就好了,然而它的输出相似于struts2??.txt,是没法直接写道咱们的XML配置文件中的。因此,咱们想到的的办法,就是在Action类中写一个方法来作转码,使它成为某个属性,因此要以get开头。而后,再用12.3.8Action注入参数(param)值一节的内容,将文件名以正常的方式设置为action类的某个属性,最后呢,再利用一个小小的param参数取值中的伎俩:${属性名},它能够直接从action类中动态获取某个属性值。好了,如今让咱们来看看第二个文件下载类FileDownloadAction2的代码:

package example;
import java.io.ByteArrayInputStream;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import com.opensymphony.xwork2.Action;
public class FileDownloadAction2 implements Action {
private String fileName;// 初始的经过param指定的文件名属性
public InputStream getInputStream() throws Exception {
return new ByteArrayInputStream("Struts 2 下载示例".getBytes());
}
public String execute() throws Exception {
return SUCCESS;
}
public void setFileName(String fileName) {
this.fileName = fileName;
}
/** 提供转换编码后的供下载用的文件名 */
public String getDownloadFileName() {
String downFileName = fileName;
try {
downFileName = new String(downFileName.getBytes(), "ISO8859-1");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return downFileName;
}
}


 这个类有两个属性,第一个是fileName,它是须要被指定的下载文件名;第二个则是动态的仅仅由getDownloadFileName()这个方法定义的属性downloadFileName,它的值随着fileName而动态变更,仅仅是把它转换成了ISO8859方式的西欧字符集。

接下来就是如何配置这个action了,这是关键的地方所在,如今配置一个新的action,名为download2,其源代码以下:

<!-- 文件下载,支持中文附件名 -->
<action name="download2" class="example.FileDownloadAction2">
<!-- 初始文件名 -->
<param name="fileName">Struts中文附件.txt</param>
<result name="success" type="stream">
<param name="contentType">text/plain</param>
<param name="inputName">inputStream</param>
<!-- 使用通过转码的文件名做为下载文件名,downloadFileName属性
对应action类中的方法 getDownloadFileName() -->
<param name="contentDisposition">attachment;filename="${downloadFileName}"</param>
<param name="bufferSize">4096</param>
</result>
</action>


 其中特殊的代码就是${downloadFileName}它的效果至关于运行的时候将action对象的属性的取值动态的填充在${}中间的部分,咱们能够认为它等价于action. getDownloadFileName()

好了,如今让咱们从新发布而后运行这个项目,键入地址:

http://localhost:8080/struts2hello/download2.action 进行访问,能够看到运行结果彻底正确,如图12.14所示。

 12.14 正确显示了文件下载名的对话框(IEFirefox

在本节的最后部分,咱们来讨论一下如何下载已经存在于当前Web应用目录下的已经存在的文件。通常的网站可能会把要下载的文件放在某个固定的目录下,例如WebRoot/download,在这个子目录下,咱们放了一个名为系统说明.doc的文件,但愿最后咱们的action可以正确的下载这个文件。要检验下载是否成功很是简单,文件内容仅仅是粗体的系统说明书这五个字,而word文件坏一个字节的话都是打不开的,因此下载后再用word打开便可检验是否成功。如今咱们建立第三个文件下载的Action类,名为example. FileDownloadAction3,其源代码清单以下所示:

package example;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import org.apache.struts2.ServletActionContext;
import com.opensymphony.xwork2.Action;
public class FileDownloadAction3 implements Action {
private String fileName;// 初始的经过param指定的文件名属性
private String inputPath;// 指定要被下载的文件路径
public InputStream getInputStream() throws Exception {
// 经过 ServletContext,也就是application 来读取数据
return ServletActionContext.getServletContext().getResourceAsStream(inputPath);
}
public String execute() throws Exception {
return SUCCESS;
}
public void setInputPath(String value) {
inputPath = value;
}
public void setFileName(String fileName) {
this.fileName = fileName;
}
/** 提供转换编码后的供下载用的文件名 */
public String getDownloadFileName() {
String downFileName = fileName;
try {
downFileName = new String(downFileName.getBytes(), "ISO8859-1");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return downFileName;
}
}


 代码中被改动的部分已经用粗斜体的方式显示出来了。首先是新加入了一个名为inputPath的属性,用来制定被下载文件的路径。接着就是ServletActionContext.getServletContext()这段代码,它的意义咱们将在12.6节详细讨论,在这里读者只须要知道它获取了当前Servlet容器的ServletContext,也就是你们常说的jsp中的application对象,而后用它来打开文件的输入流。

接着要作的就是配置action,它和刚刚配置过的download2的内容差很少,只是多了一个被下载的资源的路径属性。如今咱们在struts.xml中加入这个新的action定义:

<!-- 下载现有文件 -->
<action name="download3" class="example.FileDownloadAction3">
<param name="inputPath">/download/系统说明.doc</param>
<!-- 初始文件名 -->
<param name="fileName">系统说明.doc</param>
<result name="success" type="stream">
<param name="contentType">application/octet-stream;charset=ISO8859-1</param>
<param name="inputName">inputStream</param>
<!-- 使用通过转码的文件名做为下载文件名,downloadFileName属性
对应action类中的方法 getDownloadFileName() -->
<param name="contentDisposition">attachment;filename="${downloadFileName}"</param>
<param name="bufferSize">4096</param>
</result>
</action>


 查看粗斜体的部分,首先就是自定了被下载文件的路径,inputPath,接着就是修改了contentType为二进制方式。最后从新发布项目并运行,键入地址进行访问:http://localhost:8080/struts2hello/download3.action 。很好,能够看到文件下载对话框,保存系统说明.doc后再用word打开它,内容正确。

注意:而这种文件下载方式倒是存在安全隐患的,由于访问者若是精通Struts 2的话,它可能使用这样的带有表单参数的地址来访问:http://localhost:8080/struts2hello/download3.action?inputPath=/WEB-INF/web.xml,这样的结果就是下载后的文件内容是您系统里面的web.xml的文件的源代码,甚至还能够用这种方式来下载任何其它JSP文件的源码。这对系统安全是个很大的威胁。做为一种变通的方法,读者最好是从数据库中进行路径配置,而后把Action类中的设置inputPath的方法通通去掉,简言之就是删除这个方法定义:

public void setInputPath(String value) {
inputPath = value;
}

 而实际状况则应该成为 download3.action?fileid=1相似于这样的形式来进行。或者呢,读者能够在execute()方法中进行路径检查,若是发现有访问不属于download下面文件的代码,就一概拒绝,不给他们返回文件内容。例如,咱们能够把刚才类中的execute()方法加以改进,成为这样:

public String execute() throws Exception {
// 文件下载目录路径
String downloadDir = ServletActionContext.getServletContext().getRealPath("/download");
// 文件下载路径
String downloadFile = ServletActionContext.getServletContext().getRealPath(inputPath);
java.io.File file = new java.io.File(downloadFile);
downloadFile = file.getCanonicalPath();// 真实文件路径,去掉里面的..等信息
// 发现企图下载不在 /download 下的文件, 就显示空内容
if(!downloadFile.startsWith(downloadDir)) {
return null;
}
return SUCCESS;
}

这时候若是访问者再企图下载web.xml的内容,它只能获得一个空白页,如今访问者只能下载位于/download目录下的文件。

相关文章
相关标签/搜索