接着上一篇叙述:
二、文件上传与下载
Struts2开发的三板斧,页面jsp—配置文件struts2.xml—-还有动作类Action
文件上传前提:
form表单的method必须是post
form表单的enctype必须是multipart/form-data
提供type=”file”的上传输入域
Struts 对文件上传的支持的一些规则
1、单文件上传
开发步骤:
1)、在WEB-INF/lib下加入commons-fileupload-1.2.1.jar、commons-io-1.3.2.jar。这两个文件可以从http://commons.apache.org/下载
2)、第二步:编写upfile.jsp ,把form表的enctype设置为:“multipart/form-data“,如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="/struts-tags" prefix="s"%>
< body >
< s:actionerror />
< hr />
< s:fielderror ></ s:fielderror >
< form action = "${pageContext.request.contextPath}/upload1.action" method = "post" enctype = "multipart/form-data" > <!-- 以MIME的方式传递
-->
用户名:< input type = "text" name = "username" />< br />
靓照:< input type = "file" name = "photo" />< br />
< input type = "submit" value = "上传" />
</ form >
</ body >
|
编写错误页面error.jsp
1
2
3
|
< body >
服务器忙,一会再试。
</ body >
|
success.jsp
1
2
3
|
<body>
上传成功
</body>
|
3)、编写UploadAction1 类:在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
|
package com.itheima.actions;
import java.io.File;
import java.io.IOException;
import org.apache.commons.io.FileUtils;
import org.apache.struts2.ServletActionContext;
import com.opensymphony.xwork2.ActionSupport;
//文件上传:fileUpload拦截器完成的
public class UploadAction1 extends ActionSupport {
private String username;
private File photo; //和表单的上传字段名保持一致。类型是File类型的
private String photoFileName; //上传的文件名
private String photoContentType; //上传文件的MIME类型
//省略getter和setter方法
public String upload(){
System.out.println(photoFileName+ ":" +photoContentType);
//普通字段:
System.out.println(username);
//上传字段:上传到某个文件夹。存到应用的images目录下
String realPath = ServletActionContext.getServletContext().getRealPath( "/images" );
File directory = new File(realPath);
if (!directory.exists()){
directory.mkdirs();
}
try {
FileUtils.copyFile(photo, new File(directory, photoFileName));
return SUCCESS;
} catch (IOException e) {
e.printStackTrace();
return ERROR;
}
}
}
|
在struts.xml文件中增加如下配置
1
2
3
4
5
6
7
8
9
|
< action name = "upload1" class = "com.itheima.actions.UploadAction1" method = "upload" >
< interceptor-ref name = "defaultStack" >
< param name = "fileUpload.allowedTypes" >image/jpeg,image/png</ param >
< param name = "fileUpload.allowedExtensionsSet" >jpg,jpeg,png</ param >
</ interceptor-ref >
< result >/success.jsp</ result >
< result name = "error" >/error.jsp</ result >
< result name = "input" >/index.jsp</ result >
</ action >
|
原理分析:
a 、FileUpload 拦截器负责处理文件的上传操作, 它是默认的 defaultStack 拦截器栈的一员. 拦截器有 3 个属性可以设置.
•maximumSize: 上传文件的最大长度(以字节为单位), 默认值为 2 MB
•allowedTypes: 允许上传文件的类型, 各类型之间以逗号分隔
•allowedExtensions: 允许上传文件扩展名, 各扩展名之间以逗号分隔
可以在 struts.xml 文件中覆盖这 3 个属性
b、超出大小或非法文件的上传,会报错(转向一个input的视图)
通过:
<s:actionError/> <s:feildError/>显示错误消息的提示
c、错误消息提示改为中文版:借助国际化的消息资源文件
如果是通过配置全局默认参数引起的错误,最好用全局的消息资源文件。
struts2默认的提示资源文件:struts2-core-**.jar 的org.apache.struts2的struts-message.properties文件中。比着key值覆盖对应的value即可。
配置如下:
struts.messages.error.uploading=Error uploading: {0}
struts.messages.error.file.too.large=File too large: {0} "{1}" "{2}" {3}
struts.messages.error.content.type.not.allowed=Content-Type not allowed: {0} "{1}" "{2}" {3}
struts.messages.error.file.extension.not.allowed=File extension not allowed: {0} "{1}" "{2}" {3}
{0}:<input type=“file” name=“uploadImage”>中name属性的值
{1}:上传文件的真实名称
{2}:上传文件保存到临时目录的名称
{3}:上传文件的类型(对struts.messages.error.file.too.large是上传文件的大小)
源码:
修改显示错误的资源文件的信息
第一步:创建新的资源文件 例如fileuploadmessage.properties,放置在src下
在该资源文件中增加如下信息
struts.messages.error.uploading=上传错误: {0}
struts.messages.error.file.too.large=上传文件太大: {0} "{1}" "{2}" {3}
struts.messages.error.content.type.not.allowed=上传文件的类型不允许: {0} "{1}" "{2}" {3}
struts.messages.error.file.extension.not.allowed=上传文件的后缀名不允许: {0} "{1}" "{2}" {3}
第二步:在struts.xml文件加载该资源文件
<!-- 配置上传文件的出错信息的资源文件 -->
<constant name="struts.custom.i18n.resources" value=“cn….xxx.fileuploadmessage“/>
2、多文件上传
上传多个文件, 可以使用数组或 List,其他和单文件上传类似。
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
|
package com.itheima.actions;
import java.io.File;
import java.io.IOException;
import org.apache.commons.io.FileUtils;
import org.apache.struts2.ServletActionContext;
import com.opensymphony.xwork2.ActionSupport;
//文件上传:fileUpload拦截器完成的
public class UploadAction2 extends ActionSupport {
private String username;
private File[] photo; //和表单的上传字段名保持一致。类型是File类型的 .数组或List
private String[] photoFileName; //上传的文件名
private String[] photoContentType; //上传文件的MIME类型
public String upload(){
//上传字段:上传到某个文件夹。存到应用的images目录下
String realPath = ServletActionContext.getServletContext().getRealPath( "/images" );
File directory = new File(realPath);
if (!directory.exists()){
directory.mkdirs();
}
try {
for ( int i= 0 ;i<photo.length;i++){
FileUtils.copyFile(photo[i], new File(directory, photoFileName[i]));
}
return SUCCESS;
} catch (IOException e) {
e.printStackTrace();
return ERROR;
}
}
}
|
3、文件下载
原理:struts2提供了stream结果类型,该结果类型就是专门用于支持文件下载功能的
指定stream结果类型 需要指定一个 inputName参数,该参数指定一个输入流,提供被下载文件的入口
编码步骤:
1)、动作类DownloadAction :
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
|
package com.itheima.actions;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;
import java.net.URLEncoder;
import org.apache.commons.io.FilenameUtils;
import org.apache.struts2.ServletActionContext;
import com.opensymphony.xwork2.ActionSupport;
public class DownloadAction extends ActionSupport {
private InputStream image; //用in有问题的
private String filename; //文件名
private long filesize;
public InputStream getImage() {
return image;
}
public void setImage(InputStream image) {
this .image = image;
}
public String getFilename() {
return filename;
}
public long getFilesize() {
return filesize;
}
public String download() throws Exception{
//给image字节流赋值
String fileRealPath = ServletActionContext.getServletContext().getRealPath( "/WEB-INF/classes/霉女.jpg" );
filename = FilenameUtils.getName(fileRealPath);
//方式一:中文文件要进行URL编码
// filename = URLEncoder.encode(filename, "UTF-8");
filesize = new File(fileRealPath).length();
System.out.println(filename);
image = new FileInputStream(fileRealPath);
return SUCCESS;
}
}
|
struts.xml配置文件:主要是对stream类型的结果进行配置
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
< struts >
< constant name = "struts.devMode" value = "true" />
< constant name = "struts.ognl.allowStaticMethodAccess" value = "true" />
< action name = "download" class = "com.itheima.actions.DownloadAction" method = "download" >
< result type = "stream" >
< param name = "inputName" >image</ param > <!--动作类中InputStream的字段名,需要在Action中提供getTargetFile方法,返回inputStream-->
< param name = "contentType" >application/octet-stream</ param > <!--告诉浏览器响应头,文件的MIME格式,调用Action中的getContentType方法-->
<!-- 在struts.xml中使用OGNL表达式获取动作类中属性的值。 调用动作类中的 getFilename()-->
<!-- 中文文件名编码:方式二.使用OGNL表达式,调用URLEncode的静态方法 -->
<!-- 默认OGNL调用静态方法是不行的,需要开启一个常量开关.struts.ognl.allowStaticMethodAccess=true -->
< param name = "contentDisposition" >attachment;filename=${@java.net.URLEncoder@encode(filename,'UTF-8')}</ param > <!-- 告诉浏览器的下载方式-->
< param name = "contentLength" >${filesize}</ param >
</ result >
</ action >
</ package >
</ struts >
|
拦截器和文件上传就写到这里了,好累,不过成就感满满的。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。