OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写,它是一个开源项目。 Struts2框架使用OGNL作为默认的表达式语言。
OGNL相对其它表达式语言具有下面几大优势:
@java.lang.String@format('foo %s', 'bar')
@tutorial.MyConstant@APP_NAME
; 简单使用示例:
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="/struts-tags" prefix="s"%>
<html>
<head>
<title>演示OGNL的主要功能</title>
</head>
<body>
<br/>在Struts2中使用ONGL表达式,必须把表达式放到Struts2中的标签中,才能使用(JSP页面)<br/>
<br/>调用普通对象的普通方法<br/>
<!-- JSTL:c:out 功能和 s:property暂时任务输出内容到页面上 .property的value属性的取值就是一个OGNL表达式-->
<s:property value="'lxc'.length()"/><br/>
<s:property value="'lxc'.charAt(1)"/>
<br/>调用静态字段<br/>
<s:property value="@java.lang.Integer@MAX_VALUE"/>
</body>
</html>
ValueStack实际是一个接口,在Struts2中利用OGNL时,实际上使用的是实现了该接口的OgnlValueStack类,这个类是Struts2利用OGNL的基础 。
ValueStack(值栈): 贯穿整个 Action 的生命周期(每个 Action 类的对象实例都拥有一ValueStack 对象). 相当于一个数据的中转站. 在其中保存当前 Action 对象和其他相关对象.
1、针对用户的每次动作访问,都会创建属于自己的ValueStack对象
在 ValueStack 对象的内部有两个逻辑部分:
(1)ObjectStack: Struts 把动作和相关对象压入 ObjectStack 中–List
(2)ContextMap: Struts 把各种各样的映射关系(一些 Map 类型的对象) 压入 ContextMap 中
这些关系映射包括:
为了更好的理解可以断点执行如下代码:
Object ctx=ServletActionContext.getRequest().getAttribute("struts.valueStack");
ValueStack 类包含两个重要的属性 一个root和一个context。其中root本质上是一个ArrayList. 而context 是一个Map(更确切的说是一个OgnlContext对象)
在这个OgnlContext对象(context)中,有一个默认的顶层对象 _root
,OGNL访问context中这个默认顶层对象中的元素时,是不需要#号的,直接通过元素的名称来进行访问,
而访问其他对象时,如 request、session、attr等,则需要#号引用。
注:Struts2将ValueStack的root对象赋值给了OgnlContext 中的_root
对象,在ValueStack的root对象中,保存着调用Action的实例,因此,在页面上通过Struts2标签访问Action 的属性时,就不需要通过#号来引用
当Struts2接受一个请求时,会迅速创建ActionContext,ValueStack,action 。然后把action存放进ValueStack,所以action的实例变量可以被OGNL访问。
其中ActionContext提供了对ognl上下文对象中数据操作的方法.
ServletActionContext.getRequest().setAttribute("username", "username_request");
ServletActionContext.getServletContext().setAttribute("username", "username_application");
ServletActionContext.getContext().getSession().put("username", "username_session");
ValueStack valueStack=ServletActionContext.getContext().getValueStack();
System.out.println("valueStack "+valueStack);
valueStack.set("username", "username_valueStack");
总结 :ognl Context包含 ObjectStack属性和ContextMap属性
功能一定要知道干什么的。实现类OgnlValueStack
- getContext():返回的是一个Map<String,Object>
- *getRoot()
:返回的是一个CompoundRoot。
- CompoundRoot就是一个ArrayList的子类(就是一个List)。实现了一个栈结构
static ThreadLocal<ActionContext> actionContext = new ThreadLocal<ActionContext>();
每个线程都会有唯一的ActionContext,且一个线程中只有一个ActionContext。
最终总结:
总结:ValueStack有一个根的list叫CompundRoot和一个contextMapMap<String,String>getContext()
, 而ActionContext中存储的Map就是ValueStack中的那个Map。都是从ActionContext的那个大Map中获取的。
contextMap中
原则:OGNL表达式如果以#开头,访问的contextMap中的数据
如果不以#开头,是访问的根栈中的对象的属性
1、在动作类中Demo1Action中重写execute方法
package com.itheima.actions;
import org.apache.struts2.ServletActionContext;
import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport;
import com.opensymphony.xwork2.util.ValueStack;
public class Demo1Action extends ActionSupport{
private String username = "刘小晨";
// private String p="actionP";
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
// public String getP() {
// return p;
// }
//
// public void setP(String p) {
// this.p = p;
// }
public String execute() throws Exception {
// ServletActionContext.getRequest().setAttribute("p", "rp");//ServletRequest中的request
ServletActionContext.getRequest().getSession().setAttribute("p", "sp");//HttpSession中的session
ServletActionContext.getServletContext().setAttribute("p", "ap");//ServletContext中的application
ValueStack vs = ActionContext.getContext().getValueStack();
System.out.println(vs);
return super.execute();
}
}
配置struts.xml
<struts>
<constant name="struts.devMode" value="true" />
<package name="p1" extends="struts-default">
<action name="showContextMap" class="com.itheima.actions.Demo1Action">
<result>/contextMap.jsp</result>
</action>
</package>
</struts>
编写contextMap.jsp页面
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="/struts-tags" prefix="s"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
<title>OGNL取contextMap中的数据</title>
</head>
<body>
<br/>取contextMap中的根中(List类型的,实际类型CompoundRoot)的数据.<br/>
<s:property value="locale"/>
<s:property value="p" /><br/><!--从栈顶开始搜索Map的key或者是对象的属性-->
<s:property value="[0]" /><br/><!--不是栈顶对象,是从索引为0处砍出来的一个新集合-->
<s:property value="[0].top" /><br/><!--栈顶对象-->
<!-- value="username"的涵义:从根中的栈顶开始,对每个对象搜索他的getUsername()方法,找到为止 -->
<br/>取contextMap中的其他的数据(非根中的).<br/>
<s:property value="#request"/><br/>
动作中的:<s:property value="p"/><br/>
请求范围:<s:property value="#request.p"/><br/>
会话范围:<s:property value="#session.p"/><br/>
应用范围:<s:property value="#application.p"/><br/>
<!-- 依次从动作\页面\请求\会话范围\应用范围查找名称为p的对象 -->
attr:<s:property value="#attr.p"/><br/>
<!-- 显示出来的东西并不是contextMap中所有的东西,只是大部分 -->
<s:debug></s:debug>
</body>
</html>
数据内存结构
request:请求范围的数据。即ServletRequest中的那个Map
parameters:请求参数的数据。即request.getParameterMap得到
application:应用范围的数据。即ServletContext中的那个Map
session:会话范围的数据。即HttpSession中的那个Map
attr:也是一个Map。会从以下Map中依次搜索:request、session、application
1、 集合的投影(只输出部分属性)
<s:iterator value="allList.{name}" var="person">
<s:property/> <br>
</s:iterator>
2、 过滤条件:this 表示集合中的元素;
a.“?#
”:过滤所有符合条件的集合,如:users.{?#this.age > 19};
b.“^#
”:过滤第一个符合条件的元素,如:users.{^#this.age > 19};
c.“$#
”:过滤最后一个符合条件的元素,如:users.{$#this.age > 19} 。
<s:iterator value="allList.{?#this.age>25}" var="person">
<s:property value="name"/> xxxxxx <s:property value="age"/> <br>
</s:iterator>
3、 集合的投影和过滤
投影(过滤)操作返回的是一个集合,可以使用索引取得集合中指定的
元素,如:users.{?#this.age > 19}[0]
<s:iterator value="allList.{?#this.age>25}.{name}" var="person">
<s:property/><br>
</s:iterator>
<s:iterator value="allList.{?#this.age>25}[0]" var="person">
<s:property/><br>
</s:iterator>
编写TagDemo1Action
package com.itheima.actions;
import java.util.ArrayList;
import java.util.List;
import com.itheima.domain.Person;
import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport;
public class TagDemo1Action extends ActionSupport {
private String username="刘小晨";
private List<Person> ps = new ArrayList<Person>();
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public List<Person> getPs() {
return ps;
}
public void setPs(List<Person> ps) {
this.ps = ps;
}
public String execute() throws Exception {
//初始化一些人
ps.add(new Person("安康", "女性", 7000));
ps.add(new Person("唐诗诗", "男性", 10000));
ps.add(new Person("王卫星", "有待鉴定", 10000));
ActionContext.getContext().getValueStack().setValue("#gender", "美女");
return SUCCESS;
}
}
编写jsp页面
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="/struts-tags" prefix="s"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
<title>OGNL投影等操作</title>
</head>
<body>
<br/>--------OGNL的投影查询----------<br/>
<table border="1">
<tr>
<th>姓名</th>
</tr>
<!-- ps.{nickname} ---List<String> var=“p”引用的是一个String类型的,就代表着匿名 -->
<s:iterator value="ps.{nickname}" var="p">
<tr>
<td><s:property value="#p"/></td>
</tr>
</s:iterator>
</table>
<hr/>
<br/>--------OGNL的过滤----------<br/>
<table border="1">
<tr>
<th>姓名</th>
<th>性别</th>
<th>薪水</th>
</tr>
<s:iterator value="ps.{?#this.salary>8000}" var="p">
<tr>
<td>${p.nickname}</td>
<td><s:property value="#p.gender"/></td>
<td>${p.salary}</td>
</tr>
</s:iterator>
</table>
<table border="1">
<tr>
<th>姓名</th>
<th>性别</th>
<th>薪水</th>
</tr>
<s:iterator value="ps.{^#this.salary>8000}" var="p">
<tr>
<td>${p.nickname}</td>
<td><s:property value="#p.gender"/></td>
<td>${p.salary}</td>
</tr>
</s:iterator>
</table>
<table border="1">
<tr>
<th>姓名</th>
<th>性别</th>
<th>薪水</th>
</tr>
<s:iterator value="ps.{$#this.salary>8000}" var="p">
<tr>
<td>${p.nickname}</td>
<td><s:property value="#p.gender"/></td>
<td>${p.salary}</td>
</tr>
</s:iterator>
</table>
<s:debug></s:debug>
</body>
</html>
构造Map,如#{‘foo1’:‘bar1’, ‘foo2’:‘bar2’}。这种方式常用在给radio或select、checkbox等标签赋值上
jsp页面:
<s:radio list=“#{‘male’:‘男’,‘female’:‘女’}” name=“sex” label=“性别” />
运行结果是
<input type="radio" name="sex" id="sexmale" value="male"/>男
<input type="radio" name="sex" id="sexfemale" value="female"/>女
Action中的代码:
Map map=new HashMap();
map.put("male", "男");
map.put("female", "女");
ServletActionContext.getRequest().setAttribute("map", map);
jsp页面:
<s:property value="#request.map.male"/><br>
<s:property value="#request.map['female']"/><br>
运行结果是
男 女
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="/struts-tags" prefix="s"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
<title>利用OGNL创建List和Map(很重要)</title>
<s:head/>
</head>
<body>
<!-- 规律: 通用标签: value属性:大部分都是OGNL表达式(90%) UI标签:表单有关 value属性:大部分都不是OGNL表达式(90%) 如果要当做OGNL表达式对待:使用%{} 如果要把OGNL当做字符串对待:使用'' -->
<br/>-----创建List-------<br/>
<s:iterator value="{'a','b','c'}" var="s">
<s:property/><br/>
</s:iterator>
<br/>-----创建Map----有点像Json---<br/>
<s:iterator value="#{'a':'aaa','b':'bbb','c':'ccc'}" var="me">
<s:property value="#me.key"/>=<s:property value="#me.value"/><br/>
</s:iterator>
<br/>-----把字符串当做ONGL表达式-------<br/>
<s:set value="'昵称'" var="usernameLable"></s:set>
<s:textfield name="username" label="%{usernameLable}" value="abc" requiredLabel="true"></s:textfield>
</body>
</html>
作用:把普通字符串当做OGNL表达式来用
知识:把OGNL表达式当做普通字符串对待,请使用单引号或双引号
<param name="contentDisposition">attachment;filename=${@java.net.URLEncoder@encode(fileName,"UTF-8")}</param>
<%--
<br/>--------用OGNL来取-----------<br/>
OGNL_REQUEST:<s:property value="#request.p"/><br/>
OGNL_SESSION:<s:property value="#session.p"/><br/>
OGNL_APPLICATION:<s:property value="#application.p"/><br/>
<br/>--------用EL来取-----------<br/>
OGNL_REQUEST:${requestScope.p}<br/>
OGNL_SESSION:${sessionScope.p}<br/>
OGNL_APPLICATION:${applicationScope.p}<br/>
<br/>------------EL取数据特殊问题:Struts2对EL的功能进行了改写<br/>
${p}从request范围中取数据<br/>
--%>
${p}<br/>
<s:debug></s:debug>