struts2框架
struts2是一种基于MVC模式的框架,是在struts1的基础上融合了xwork的功能。
struts2框架预处理了一些功能:
>请求数据自动封装,
>文件上传的功能
>对国际化功能的简化
>数据校验功能
使用struts2框架开发的流程:
1.引入jar文件
>commons-fileupload-1.2.2.jar 【文件上传相关包】
>commons-io-2.0.1.jar
>struts2-core-2.3.4.1.jar 【struts2核心功能包】
>xwork-core-2.3.4.1.jar 【Xwork核心包】
>ognl-3.0.5.jar 【Ognl表达式功能支持表】
>commons-lang3-3.1.jar 【struts对java.lang包的扩展】
>freemarker-2.3.19.jar 【struts的标签模板库jar文件】
>javassist-3.11.0.GA.jar 【struts对字节码的处理相关jar】
2.配置web.xml
Tomcat启动时,会加载所有项目的web.xml,通过web.xml中引入过滤器,而struts的核心功能
的初始化,是通过过滤器完成的
<!--引入核心过滤器-->
<filter>
<filter-name>struts2</filter-name>
<filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
StrutsPrepareAndExecuteFilter就是核心过滤器,位于struts2的核心功能包中,使用struts版本不同,核心过滤器类是不一样的
3.配置struts.xml
struts2的执行流程
1.服务器启动
>加载web.xml
>创建struts核心过滤器对象,执行filter-->init()
·struts-default.xml, 核心功能的初始化
·struts-plugin.xml, struts相关插件
·struts.xml 用户编写的配置文件
2.用户访问
>用户访问action,服务器根据路径名称,找对应的action配置,创建action对象
>执行默认拦截器栈中定义的18个拦截器
>执行action的业务处理方法
struts-default.xml详解
1.位于struts2-core-2.3.jar包中,
2.bean节点指定了struts在运行时候创建的对象类型
3.指定struts-default包,用户写的struts.xml文件中package一定要继承此包,struts-default包定义了:
>跳转的结果类型:dispatcher,redirect,redirectAction,stream
>定义了所有拦截器,一共32个拦截器,为了拦截器引用方便,可以通过定义栈的方式引用拦截器,默认的栈中包含了初始化18个拦截器
>默认执行的拦截器栈、默认执行的action
拦截器和过滤器比较:
相似:功能相似
区别: 过滤器,拦截器所有资源都可以; (/index.jsp/servlet/img/css/js)
拦截器,只拦截action请求。
拦截器是struts的概念,只能在struts中用。
过滤器是servlet的概念,可以在struts项目、servlet项目用
注意:拦截器什么时候执行,先执行Action类创建,还是先执行拦截器
答:拦截器在访问时执行,先创建Action类对象,再按顺序执行18个拦截器,最后执行Action类的业务处理方法
Action开发
Action开发有三种方式:
1.继承ActionSupport类,如果用struts的数据校验功能,必须继承此类
2.实现Action接口,重写execute方法
3.不继承任何类,不实现任何接口
struts中路径匹配原则:
localhost:访问到哪一台机器
8080:找到Tomcat
mystruts:找到项目名
/user/a/b:查找是否有此名称空间,没有则向下
/user/a:查找是否有此名称空间,没有则向下
/user:查找是否有此名称空间,没有则向下
/:查找是否有此名称空间,没有则报错
例如:<action name="login" class="..." method="..."></action>,项目名称后面可以有/a/b这些无用的分层,只要最后为login以及名称空间正确照样能访问
struts中的常量
>struts中的常量定义了默认访问后缀等配置,文件名为default.propertities
>位于struts核心包中
>能够在struts.xml中通过<constant name="key" value="value"></coonstant>修改默认常量配置
struts中对数据操作(三种方式)
1.直接拿到servletAPI,进行操作,核心类:ServletActionContext提供的静态方法
HttpServletRequest request = ServletActionContext.getRequest();
HttpSession session = request.getSession();
ServletContext application = ServletActionContext.getServletContext();
2.通过ActionContext类获取(代表request,session,application)map
ActionContext ac = ActionContext.getContext();
//struts对HttpServletRequest进行了封装,封装成一个map //拿到表示request对象的map
Map<String, Object> request = ac.getContextMap(); //拿到session对象的map
Map<String, Object> session = ac.getSession(); //拿到表示ServletContext对象的map
Map<String, Object> application = ac.getApplication();
3.实现接口的方式:(requestAware/sessionAware/applicationAware)
public class DataAction2 extends ActionSupport implements RequestAware,
SessionAware, ApplicationAware {
Map<String, Object> request;
Map<String, Object> session;
Map<String, Object> application; // struts运行时,会把代表request的map对象注入
@Override
public void setRequest(Map<String, Object> request) {
this.request = request;
} // struts运行时,会把代表session的map对象注入
@Override
public void setSession(Map<String, Object> session) {
this.session = session;
} // struts运行时,会把代表application的map对象注入
@Override
public void setApplication(Map<String, Object> application) {
this.application = application;
} @Override
public String execute() throws Exception {
request.put("request_data", "request_actionAware");
session.put("session_data", "session_actionAware");
application.put("application_data", "application_actionAware"); return SUCCESS;
}
}
区别:第二种方式由于不用引进servlet包,是解耦的方式实现对数据的操作,所以推荐使用第二种方式
但是,第二种方式每个事务方法都要获取ActionContext,因为ActionContext可以获取struts的数据或者对象,而这一过程
是通过拦截器完成的,而拦截器在创建Action对象之后运行,所以无法把ActionContext对象在全局变量中获取,第二种和第三种
方式原理一样,不过第三种方式通过接口则解决了第二种方式的不足,但实现起来较麻烦,业务方法比较多的时候应该优先考虑。
第一种方式可以在二三种方式无法实现需求是使用比如获取request.getContextPath()等。
struts2中请求数据自动封装(两种方式)
实现原理:参数拦截器,用户访问时,创建了Action对象,因此拦截器能够拿到action对象和属性
<interceptor name="params" class="com.opensymphony.xwork2.interceptor.ParametersInterceptor"/>
1.jsp表单数据填充到action中的属性,需要给出属性的set方法
2.jsp表单数据填充到action的对象中的属性:一定要给出对象的set和get方法
// 第一种方式
/*
* private String username;// 请求数据封装,必须给出set方法
* private String password;
* private int age;
* private Date birth;
*
* public void setUsername(String username) { this.username = username; }
*
* public void setPassword(String password) { this.password = password; }
*
* public void setAge(int age) { this.age = age; }
*
* public void setBirth(Date birth) { this.birth = birth; }
*/ // 第二种方式
private User user; public void setUser(User user) {
this.user = user;
} // 处理注册请求
public String regist() {
System.out.println("username:" + user.getUsername() + ",password:"
+ user.getPassword() + ",age:" + user.getAge() + ",birth:"
+ user.getBirth());
return SUCCESS;
}
数据类型转换
struts中jsp提交的数据,会自动转换为action中的属性的类型
对于基本类型以及日期类型会自动转换,日期类型只支持yyyy-MM-dd格式
如果是其他格式,需要自定义类型转换器
自定义类型转换器:
struts中转换器API
|--TypeConverter 转换器接口
|--DefaultTypeConverter 默认类型转换器(yyyy-MM-dd就是在这里定义的)
|--StrutsTypeConverter 用户编写的类型转换器,继承此类即可
局部转换器开发步骤:
1.写转换器类,重写父类的抽象方法
2.配置转换器类(告诉struts使用自定义的转换器类)
-->在同包的action目录下,新建一个properties文件
-->命名规则:ActionClassName-conversion.properties
举例:cn.electhuang.d_type/UserAction-conversion.properties
-->配置文件内容:user.birth=转换器类全路径(cn.electhuang.d_type.MyConverter)
总结:转换器(转换器类和配置文件)不能给其他Action用
全局类型转换器
需要写一个转换器给所有的action用
配置:
-->在src目录下,新建一个properties文件
-->命名规则:xwork-conversion.properties
-->内容:类型全名=转换器类(java.util.Date=cn.electhuang.d_type.MyConverter)
转换器类代码示例:
public class MyConverter extends StrutsTypeConverter {
// 要求支持三种Date格式
// 先定义格式
DateFormat[] df = { new SimpleDateFormat("yyyy-MM-dd"),
new SimpleDateFormat("yyyyMMdd"),
new SimpleDateFormat("yyyy年MM月dd日") }; /**
* 把String转换为指定类型
*
* @param context
* 当前上下文环境
* @param values
* jsp提交的字符串值
* @param toClass
* 要转换的目标类型
* @return the converted object
*/
@Override
public Object convertFromString(Map context, String[] values, Class toClass) { if (values == null || values.length == 0) {
return null;
}
System.out.println("1111");
if (Date.class != toClass) {
return null;
}
System.out.println("2222");
for (int i = 0; i < df.length; i++) {
try {
return df[i].parse(values[0]);
} catch (ParseException e) {
continue;
}
}
System.out.println("3333");
return null;
} @Override
public String convertToString(Map context, Object o) {
return null;
} }
struts文件上传
1.回顾采用servlet文件上传
前台:
-->提交方式必须为POST
-->表单类型:multipart/form-data
-->input type="file"
后台:
-->Apache提供的FileUpload组件
-->核心类:
①FileItemFactory FileItem的工厂
②ServletFileUpload servlet中文件上传的核心类
③FileItem 封装了上传的表单文件项的信息
总结:文件上传使用比较多,但处理比较麻烦!
2.struts的文件上传
文件上传拦截器帮助完成文件上传的功能
<interceptor name="fileUpload" class="org.apache.struts2.interceptor.FileUploadInterceptor"/>
代码示例:
public class FileUpload extends ActionSupport {
private File file1;// 对应表单的name
private String File1FileName;// 文件名
private String file1ContentType;// 文件类型 public void setFile1(File file1) {
this.file1 = file1;
} public void setFile1FileName(String file1FileName) {
File1FileName = file1FileName;
} public void setFile1ContentType(String file1ContentType) {
this.file1ContentType = file1ContentType;
} @Override
public String execute() throws Exception {
// 拿到上传文件进行处理
// 把文件上传到upload目录
// 获取上传的目录路径
String path = ServletActionContext.getServletContext().getRealPath(
"/upload"); // 创建目标文件对象
File destFile = new File(path, File1FileName); // 把上传的文件拷贝到目标文件对象中
FileUtils.copyFile(file1, destFile); return SUCCESS;
} }
重要:文件上传细节处理
-->文件大小限制:struts2默认支持上传最大是2M,可通过常量修改
当文件上传出现错误时,struts内部会返回input视图(错误视图),所以需要在struts.xml中配置input视图对应的错误页面
<!-- 4. 修改上传文件的最大大小为30M -->
<constant name="struts.multipart.maxSize" value="31457280"/>
-->限制上传文件的运行类型
拦截器注入参数从而限制文件上传类型
<!-- 限制上传文件的类型 -->
<interceptor-ref name="defaultStack">
<!-- 限制文件扩展名 -->
<param name="fileUpload.allowedExtensions">txt,jpg</param>
</interceptor-ref>
struts文件下载(2种方式)
1.通过response对象向浏览器写入字节流数据,设置下载的响应头
2.struts的方式
代码示例:
public class DownAction extends ActionSupport {
// 显示所有要下载的文件列表
public String list() {
// 得到upload目录
String path = ServletActionContext.getServletContext().getRealPath(
"/upload");
// 创建目录对象
File file = new File(path);
// 得到所有文件名
String[] fileNames = file.list();
// 保存
ActionContext ac = ActionContext.getContext();
Map<String, Object> request = ac.getContextMap();
request.put("fileNames", fileNames);
return "list";
} /*
* 文件下载
*/
// 1.获取要下载的文件名
private String fileName; public void setFileName(String fileName) {
try {
// 处理传入参数中文乱码问题
fileName = new String(fileName.getBytes("ISO8859-1"), "utf-8");
} catch (UnsupportedEncodingException e) {
throw new RuntimeException(e);
}
// 把处理好的文件名赋给fileName
this.fileName = fileName;
} // 2.下载提交的业务方法,需要在struts.xml中配置返回stream
public String down() {
return "download";
} // 3.返回流
public InputStream getAttrInputStream() { return ServletActionContext.getServletContext().getResourceAsStream(
"/upload/" + fileName);
} // 下载显示的文件名
public String getDownFileName() {
// 需要中文编码
try {
fileName = URLEncoder.encode(fileName, "utf-8");
} catch (UnsupportedEncodingException e) {
throw new RuntimeException(e);
}
return fileName;
}
}
配置文件配置方法:
<struts>
<package name="fileupload_" namespace="/" extends="struts-default" abstract="flase">
<!-- action的名称不要用FileUpload -->
<action name="fileUpload_" class="cn.electhuang.e_fileupload.FileUpload" method="execute">
<result name="success">/e/success.jsp</result> <!-- 配置错误视图 -->
<result name="input">/e/error.jsp</result> <!-- 限制上传文件的类型 -->
<interceptor-ref name="defaultStack">
<!-- 限制文件扩展名 -->
<param name="fileUpload.allowedExtensions">txt,jpg</param>
</interceptor-ref>
</action> <action name="down_*" class="cn.electhuang.e_fileupload.DownAction" method="{1}">
<!-- 列表展示 -->
<result name="list">/e/list.jsp</result> <!-- 下载操作 -->
<result name="download" type="stream">
<!-- 允许下载的文件类型,指定为所有的二进制文件类型 -->
<param name="contentType">application/octet-stream</param> <!-- 对应action中的属性,返回流的属性 ,其实就是找到getArrrInputStream方法-->
<param name="inputName">attrInputStream</param> <!-- 下载头,包括浏览器显示的文件,其实就是找到getDownFileName方法 -->
<param name="contentDisposition">attachment;filename=${downFileName}</param> <!-- 缓冲区大小设置 -->
<param name="bufferSize">1024</param>
</result>
</action> </package>
</struts>