Struts2 框架的值栈
1. OGNL 表达式
1.1 概述
- OGNL(Object Graphic Navigation Language),即对象图导航语言;
- 所谓对象图,即以任意一个对象为根,通过OGNL可以访问与这个对象关联的其它对象;
- 通过OGNL表达式,可以存取对象的任意属性,调用对象的方法,遍历整个对象的结构图等功能;
1.2 特点
- OGNL 是一种比 EL 表达式强大很多倍的语言;
- xwork 提供 OGNL 表达式;
ognl-3.0.5
包;- Struts2 框架使用 OGNL 作为默认的表达式语言;
1.3 OGNL 提供五大类功能
- 支持对象方法调用;
- 支持类静态的方法调用和值访问;
- 访问 OGNL 上下文(OGNL context) 和 ActionContext;
- 支持赋值操作和表达式串联;
- 操作集合对象;
// 示例:
public void fun throws OgnlException{
// 获取上下文对象
OgnlContext context = new OgnlContext();
// 获取到根对象
Object root = context.getRoot();
// 存入值
context.put("name","张三");
// 通过 OGNL 表达式获取值
Object value = Ognl.getValue("#name",context,root);
System.out.println(value);
}
2. 在 Struts2 框架中使用 OGNL 表达式
2.1 Struts2 引入 OGNL 表达式,主要是在 JSP 页面中获取值栈中的值;
// 使用 OGNL 表达式的步骤
// 1. 在 JSP 页面中引入 Struts2 的标签库
<%@ taglib prefix="s" uri="/struts-tags" %>
// 2. 使用 Struts2 提供的 property 标签, 从值栈中获取值
// <s:property value="OGNL表达式"/>
// jsp 页面输出 hello 字符串的长度
<s:property value="'hello'.length()"/>
3. 值栈
3.1 值栈的概述
- 值栈就相当于 Struts2 框架的数据中转站, Web 层向值栈中存入一些数据, JSP 页面从值栈中获取数据;
ValueStack
是 Struts2 提供的一个接口; OGNL 表达式通过该接口的实现类OgnlValueStack(值栈对象)
从值栈中获取数据.- Action 是多例的,每发起一个请求,就会创建一个Action实例,一个
ActionContext
对象, 一个ValueStack
对象; - 每个 Action 实例都有一个 ValueStack 对象,其中保存当前 Action 对象和其他相关对象;
- Struts2 框架将
ValueStack
对象保存在 request 域中,名称为struts.valueStack
ValueStack vs = (ValueStack)request.getAttribute("struts.valueStack");
3.2 值栈的内部结构
3.2.1 值栈由两部分组成
- ValueStack 中存在 root 属性(CompoundRoot), context 属性(OgnlContext);
CompoundRoot
就是 ArrayList;OgnlContest
就是 Map;
root
: Struts 把动作和相关对象压入到ObjectStack
中;context
: Struts 把各种各样的映射关系(一些 Map 类型的对象)压入到ContextMap
中;- Struts2 会默认把下面这些映射压入到
ContextMap
中parameters
: 该 Map 中包含当前请求的请求参数;request
: 该 Map 中包含当前 request 对象中的所有属性;session
: 该 Map 中包含当前 session 对象中的所有属性;application
: 该 Map 中包含将当前 application 对象中的所有属性;attr
: 该 Map 按如下顺序来检索某个属性, request, session, application;- 注意:
request
表示 Map 集合中的 key值, value 值其实也是一个 Map 集合;
3.2.2 OGNL 表达式访问值栈中的数据
- 访问
root
中的数据时,不需要#
; - 访问
context
中的数据时,必须写#
; - 操作值栈默认指操作
root
元素;
3.3 值栈的创建和 ActionContext 对象的关系
- 值栈对象(ValueStack)是请求时创建的;
- ActionContext 是绑定在当前的线程上,每个拦截器或者 Action 中获取到的 ActionContext 是同一个;
- ActionContext 中存在一个 Map 集合,该 Map 集合和 ValueStack 的 context 是同一个地址;
- ActionContext 中可以获取到 ValueStack 的引用;换句话说,以后使用 ActionContext 对象来获取值栈对象;
ValueStack vs = ActionContext.getContext().getValueStack();
3.4 向值栈中保存数据(主要针对 root 栈)
valueStack.push(Object obj)
- push 方法的底层调用 root 对象的 push 方法(把元素添加到栈顶);
valueStack.set(String key, Object obj)
- 向栈顶压入 map 集合,把 key 和 obj 存入到 map 集合中;
- 首先获取栈顶中的值,查看是否是 map 集合, 如果是,直接把 key 和 obj 存入到该 map 集合中;
如果不是 map 集合,会先创建 HashMap 集合,然后存入 key 和 value 的值.
- 向
root
栈中存入对象,优先使用push
方法; 向root
栈中存入集合,优先使用set
方法.
3.5 使用OGNL表达式从值栈中获取数据
- JSP 页面中写入
<s:debug></s:debug>
,可以查看valueStack
中存入的数据;
// 第一种情况: 从 root 栈中取值
// 如果向值栈中保存字符串
vs.push("张三");
// 在 JSP 页面获取栈顶的值
<s:property value="[0].top"/>
// 如果向值栈中保存 map 集合
vs.set("msg","测试代码");
// 在 JSP 页面获取栈顶的值
<s:property value="[0].top.msg"/>
// 如果向值栈中保存 user 对象
vs.push(user);
// 获取栈顶的值
<s:property value="[0].top.username"/>
<s:property value="[0].top.password"/>
// 注意: [0].top 关键字可以省略
// 因此可以简写为
<s:property value="username"/>
// 向值栈中保存 map 集合
vs.set("user",user);
// 获取栈顶的值
<s:property value="[0].top.user.username"/>
<s:property value="[0].top.user.password"/>
// 示例:
public class ValueStackAction extends ActionSupport{
// ValueStackAction 中的属性
private User user = new User("张三",22);
public User getUser(){
return user;
}
public void setUser(User user){
this.user = user;
}
public String add() throws Exception{
// 获取到值栈对象,要先获取 ActionContext 对象
ValueStack vs = ActionContext.getContext().getValueStack();
User u2 = new User("李四",22);
// 向值栈中存入 user 对象
vs.set("user",u2);
}
}
// JSP 页面中取值
// 从栈顶开始查找user, 找到之后,获取 user 的属性
<s:property value="user.username"/> 获取到的 username 为"李四"
// [1].top 获取 ValueStackAction 对象中的属性 user
<s:property value="[1].top.user.username"/> 获取到的 username 为"张三"
// 从JSP页面获取集合中的数据
public class ValueStackAction extends ActionSupport{
public String add() throws Exception{
// 获取值栈
ValueStack vs = ActionContext.getContext().getValueStack();
// 向值栈中存入 list 集合
List<User> ulist = new ArrayList<User>();
ulist.add(new User("张三",22));
ulist.add(new User("李四",25));
ulist.add(new User("王五",33));
vs.push(ulist);
}
}
// JSP页面取值
<s:property value="[0].top[0].username"/> // 获取到"张三"
// 如果 vs.set("uslist",ulist);
// JSP 页面取值
<s:property value="ulist[0].username"/> // 获取到"张三"
// JSP 页面中使用迭代标签<s:iterator>, 该标签中的属性
// * value: 要迭代的集合,需要从值栈中获取;
// * var: 迭代过程中,遍历的对象;
// * 如果有 var, 会把迭代产生的对象默认压入到 context 栈中;从context栈中取值,要加"#"号
// * 如果没有 var, 会把迭代产生的对象默认压入到 root 栈中;
// 带有 var 属性
<s:iterator value="ulist" var="aaa">
<s:property value="#aaa.username"/>
<s:property value="#aaa.age"/>
</s:iterator>
// 没有 var 属性
<s:iterator value="ulist">
<s:property value="username"/>
<s:property value="age"/>
</s:iterator>
// 第二种情况: 从 context 栈中取值,底层已经封装了 request,session,application等对象
// 向 context 栈中存入值
HttpServletRequest request = ServletActionContext.getRequest();
request.setAttribute("msg","张三");
// JSP 页面获取值
<s:property value="#request.msg"/>
3.6 使用EL表达式从值栈中获取数据
// 接上面存入的值
// 在JSP页面使用EL表达是和 JSTL 标签
// 底层使用的是 装饰者模式, 增强了 getAttribute() 方法
<c:forEach items="${ulist}" var="user">
${user.username}<br/>
${user.age}
</c:forEach>
4. OGNL 表达式的特殊符号
4.1 "#"的用法
- 获取 context 栈中的数据:
<s:property value="#request.name"/>
- 构建一个 map 集合:
<s:radio name="sex" list="#{'0':'男','1':'女'}"></s:radio>
4.2 "$"的用法
- 在配置文件中可以使用 OGNL 表达式,例如: 文件下载的配置文件
<action name="download" class="cn.itcast.demo.DownloadAction">
<result name="success" type="stream">
<param name="contentType">${contentType}</param>
<param name="contentDisposition">attachment;filename=${downFilename}</param>
</result>
</action>
参考资料