前言

Struts2的文件下载Action与普通的Action并没有什么太大的不同,仅仅是该Action需要提供一个返回InputStream流的方法,该输入流代表了被下载文件的入口,同时在配置文件中配置Action的result类型为stream。

文件下载

Action类的实现:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
public class FileDownloadAction extends ActionSupport {

private static final long serialVersionUID = 1L;

// 文件的Mime类型
private String contentType;

// 要下载的文件名称
private String fileName;

private File[] files;

public String list() throws Exception {
File f = new File(getDownloadFilePath());
// 获取目录下保存的文件
files = f.listFiles();
return "list";
}

/*
* 定义一个返回InputStream的方法 该方法将被作为下载文件的入口, 且需要配置Stream类型结果时指定inputName参数
* inputName参数的值就是该方法去掉get前缀,首字母小写的字符串
*/
public InputStream getDownloadFile() throws Exception {
String filePath = getDownloadFilePath() + "/" + fileName;

InputStream is = new FileInputStream(new File(filePath));
return is;
}

//设置下载文件的名称
public String getDownloadFileName() {
String downFileName = fileName;

// get userAgent
String userAgent = ServletActionContext.getRequest().getHeader("User-Agent");

try {
// Edge Browser or IE Browser
if (userAgent.contains("Edge") || userAgent.contains("Trident") || userAgent.contains("MSIE")) {
downFileName = java.net.URLEncoder.encode(downFileName, "UTF-8").replace("+", "%20");
} else {
downFileName = new String(downFileName.getBytes("UTF-8"), "ISO-8859-1");
}
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return downFileName;
}

public String getDownloadFilePath() throws Exception {
String filePath = getClass().getResource("/").toURI().getPath();
filePath = filePath.replace("WEB-INF/classes/", "download");

return filePath;
}

public File[] getFiles() {
return files;
}

public void setFiles(File[] files) {
this.files = files;
}

public void setFileName(String fileName) throws UnsupportedEncodingException{
//下载文件名如果包含中文需要转码,避免出现乱码
fileName = URLDecoder.decode(fileName,"UTF-8");
this.fileName = fileName;
}

public String getFileName() {
return fileName;
}

public String getContentType() {
return contentType;
}

public void setContentType(String contentType) {
this.contentType = contentType;
}
}

Struts.xml的配置

配置下载文件的Action时,需要配置一个类型为stream的结果,需要指定如下4个属性。

  • contentType: 指定被下载文件的文件类型
  • inputName: 指定被下载文件的入口输入流
  • contentDisposition: 指定下载的文件名
  • bufferSize: 指定下载文件时的缓冲大小
1
2
3
4
5
6
7
8
9
10
11
12
<action name="download" class="com.learn.FileDownloadAction.FileDownloadAction">
<result type="stream">
<!-- 指定下载文件的MIME类型 -->
<param name="contentType">${contentType}</param>
<!-- 下载的文件数据存放的方法,该方法返回一个InputStream 例如取值为inputStream的属性需要编写getInputStream()方法 -->
<param name="inputName">downloadFile</param>
<!--下载时,客户端显示的下载的文件名 -->
<param name="contentDisposition">attachment;filename=${downloadFileName}</param>
<!-- 数据的缓冲大小 -->
<param name="bufferSize">4096</param>
</result>
</action>

文件下载取消时出现异常

在struts2中使用result里type=”stream”的结果类型时,当在“下载提示窗口”中点击“取消按钮”时,总是报出“java.lang.IllegalStateException”异常。

异常原因分析:

  1. 配置其中result标签下的各个参数
  2. 从服务器中获取输入流,并同时与客户端建立输出流(服务器与客户端链接通过Socket进行连接)
  3. 当点击“保存”或“打开”时,开始传输数据。如果点击“取消”,关闭所有的流。这里要注意的是,但是实际发现Socket并没有断开!并且流也没有关闭!这一点非常重要!所以在JSP容器通过Response获取输出流之前,前面的流并没有关闭,所以会造成该异常的报出。

异常解决办法:

  1. 下载插件struts2-sunspoter-stream-1.0.jar,并复制在/WEB-INF/lib下。
    CSDN下载地址:struts2-sunspoter-stream-1.0.jar

  2. 在Struts.xml 中配置:

1
2
3
<result-types>
<result-type name="streamx" class="com.sunspoter.lib.web.struts2.dispatcher.StreamResultX" />
</result-types>

3.在下载的Action中配置result的类型为streamx:

1
2
3
4
5
6
7
8
9
10
<action name="download" class="com.learn.FileDownloadAction.FileDownloadAction">
<result type="streamx">
<!-- 下载的文件数据存放的方法,该方法返回一个InputStream 例如取值为inputStream的属性需要编写getInputStream()方法 -->
<param name="inputName">downloadFile</param>
<!--下载时,客户端显示的下载的文件名 -->
<param name="contentDisposition">attachment;fileName=${downloadFileName}</param>
<!-- 数据的缓冲大小 -->
<param name="bufferSize">4096</param>
</result>
</action>

PS(附言):Demo下载地址

最后更新: 2018年09月05日 11:19

× 请我吃糖~
打赏二维码