一、ServletAPI 获取方式
- 第一种:通过ServletActionContext获取 【推荐使用】
- 第二种:通过注入方式:先让Action实现两个接口ServletRequestAware,ServletResponseAware
第二种注入方式原理是因为struts有个拦截器
<interceptor name="servletConfig" class="org.apache.struts2.interceptor.ServletConfigInterceptor"/>
来实现的查看ServletConfigIntercepto的源码
[图片上传失败...(image-b4f96a-1554098535653)]
二、Result元素
作用
为动作指定结果视图
属性
- name:逻辑视图的名称,对应着动作方法的返回值。默认值是success
- type:结果类型,指的就是用什么方式转到定义的页面。默认是dispatcher转发。
result中type的取值(四种类型)
- dispatcher (默认值)使用请求转发,转向一个页面。
- redirect 使用重定向,转向一个页面。
- chain 转发到另一个相同名称空间的动作转发到不同名称空间的动作
- redirectAction 重定向到另一个相同名称空间的动作重定向到不同名称空间的动作
其他类型
<result-types>
<result-type name="chain" class="com.opensymphony.xwork2.ActionChainResult"/>
<result-type name="dispatcher" class="org.apache.struts2.dispatcher.ServletDispatcherResult" default="true"/>
<result-type name="freemarker" class="org.apache.struts2.views.freemarker.FreemarkerResult"/>
<result-type name="httpheader" class="org.apache.struts2.dispatcher.HttpHeaderResult"/>
<result-type name="redirect" class="org.apache.struts2.dispatcher.ServletRedirectResult"/>
<result-type name="redirectAction" class="org.apache.struts2.dispatcher.ServletActionRedirectResult"/>
<result-type name="stream" class="org.apache.struts2.dispatcher.StreamResult"/>
<result-type name="velocity" class="org.apache.struts2.dispatcher.VelocityResult"/>
<result-type name="xslt" class="org.apache.struts2.views.xslt.XSLTResult"/>
<result-type name="plainText" class="org.apache.struts2.dispatcher.PlainTextResult" />
<result-type name="postback" class="org.apache.struts2.dispatcher.PostbackResult" />
</result-types>
result元素中param子元素
- 在转发或者重定向到不同包下的动作时,都用到了result元素的子元素param。
- param元素的作用:依赖注入(Dependence Injection)思想
- 我们通过struts-default.xml中的resultTypes元素中配置可以看出,每个结果类型视图其实都是靠一个类来实现的。
- 而param元素就是将配置的参数,注入到该类中。
- 调用的是对应类的setter方法进行注入的。
自定义结果类型
- 其实结果类型就是一个类,这些类都实现com.opensymphony.xwork2.Result接口。
- 或者继承自该接口的实现类org.apache.struts2.dispatcher.StrutsResultSupport。
- 这些类都有一个doExecute方法,用于执行结果视图。【查看源码各种结果类型的类结构】
- struts的内部实现就是Servlet
三、拦截器
简介
- Struts2中的很多功能都是由拦截器完成的。比如:servletConfig,staticParam,params,modelDriven等等。
- Struts默认的拦截器配置在struts-default.xml文件中
- 拦截器是AOP编程思想的一种应用形式。
执行顺序
[图片上传失败...(image-fad3dd-1554098535653)]
自定义拦截器
方式一
可以看到,struts默认的拦截器都是实现了AbstractIntercepter,所以继承这个抽象类并且实现它的intercept方法即可,
public class MyInterceptor extends AbstractInterceptor {
@Override
public String intercept(ActionInvocation actionInvocation) throws Exception {
System.out.println("开始。。。。");
String invoke = actionInvocation.invoke();
System.out.println("结束。。。");
return invoke;
}
}
invoke默认值是success;也就是结果视图;
方式二
继承AbstractIntercepter的子类MethodFilterInterceptor;实现其doIntercept方法
这种方式适用于对方法拦截进行配置
配置拦截器
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts PUBLIC
"-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
"http://struts.apache.org/dtds/struts-2.0.dtd">
<struts>
<package name="p1" extends="struts-default" namespace="/user">
<result-types>
<result-type name="chptcha" class="com.unclezs.web.result.ChptchaResult"></result-type>
</result-types>
<interceptors>
<interceptor name="myinter" class="com.unclezs.web.interceptor.MyInterceptor"></interceptor>
</interceptors>
<action name="code">
<interceptor-ref name="myinter"></interceptor-ref>
<result type="chptcha">
<param name="width">200</param>
<param name="height">40</param>
</result>
</action>
<action name="regist" class="com.unclezs.web.action.UserRegist" method="regist">
<result name="success" type="dispatcher">/success.jsp</result>
</action>
</package>
</struts>
四、文件上传
- Struts也提供了内置标签用于文件上传<s:file>,我们称为文件选择域与html的标签<input tyle=”file”>是一样用的
- 文件上传的必要前提条件,表单必须是post方法,enctype类型必须为multipart/form-data
JSP页面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@taglib prefix="s" uri="/struts-tags" %>
<s:head></s:head>
<html>
<head>
<title>文件上传</title>
</head>
<body>
<s:form action="user/upload" enctype="multipart/form-data" method="POST">
<s:file label="文本" name="file"></s:file>
<s:fielderror></s:fielderror>
<s:submit value="上传"></s:submit>
</s:form>
</body>
</html>
Action类及XML配置
[图片上传失败...(image-b72873-1554098535653)]
五、OGNL表达式
简介
- OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写,它是一个单独的开源项目。 Struts2框架使用OGNL作为默认的表达式语言。
- OGNL是struts2整合的一个开源项目,所以在struts2中,要想使用OGNL表达式,必须使用Struts2标签库OGNL相当 于EL表示式,从作用域取数据
OGNL简单使用
s:property类型于JSP的表达式,把value的值直接输出到页面
jsp表达式:【<jsp:setProperty property="" name=""/>】
value的值就是一个OGNL表达式,它不是一个字符串
如果想把value的值当字符串输出,加单引号即可
功能
- 访问对象方法
- 访问静态方法
- 访问静态属性
默认OGNL是禁止静态方法访问的,在default.properties中有个struts.ognl.allowStaticMethodAccess=false,如想使用,需要在struts.xml文件中配置允许访问静态方法 - 封装list数据
- 封装Map数据
- struts.xml中写OGNL
六、contextMap
动作类的生命周期
strtus2动作类是多例的,每次动作访问,动作类都会实例化。所以是线程安全的。
与Struts1的区别是,struts1的动作类是单例的。
请求数据的存放
- 在每次动作执行前,核心控制器StrutsPrepareAndExecuteFilter都会创建一个ActionContext和ValueStack对象。且每次动作访问都会创建
- 这两个对象存储了整个动作访问期间用到的数据。
- 并且把数据绑定到了线程局部变量(ThreadLocal)上了。所以是线程安全的。
存储数据
contextMap=ActionContext中存放的主要内容
Key------Value------说明
- value stack (root)------java.util.List------没有root这个key。它是一个list。
- application------java.util.Map<String,Object>------ServletContext中的所有属性。
- session------java.util.Map<String,Object>------HttpSession中的所有属性。
- request------java.util.Map<String,Object>------ServletRequest中的所有属性。
- parameters------java.util.Map------参数
- attr------java.util.Map------把页面、请求、会话、应用范围内的所有属性放到一起。
注意:
除了value stack之外,全是map,而contextMap也是一个map。
其实就是Map中又封装的Map。(很像dbutils中KeyedHandler封装数据的结构,只是封装数据的结构)
使用s:debug查看contextMap的数据