一、OGNL概述
1.1 什么是OGNL
OGNL的全称是对象图导航语言( object-graph Navigation Language),它是一种功能强大的开源表达式语言,使用这种表达式语言,可以通过某种表达式语法,存取Java对象的任意属性,调用Java对象的方法,同时能够自动实现必要的类型转换。如果把表达式看作是一个带有语义的字符串,那么OGNL无疑成为了这个语义字符串与Java对象之间沟通的桥梁。
1.2 OGNL的作用
Struts2默认的表达式语言就是OGNL,它具有以下特点:
支持对象方法调用。例如: objName. methodName( )。
支持类静态方法调用和值访问,表达式的格式为@[类全名(包括包路径)]@[ 方法名|值名 ]。例如:@ java.lang.String@format("foo%s","bar")。
支持赋值操作和表达式串联,例如: price=100, discount=0.8, calculatePrice( ),在方法中进行乘法计算会返回80
访问OGNL上下文( OGNL context)和 ActionContext。
操作集合对象。
1.3 OGNL的要素
了解了什么是OGNL及其特点后,接下来,分析一下OGNL的结构。OGNL的操作实际上就是围绕着OGNL结构的三个要素而进行的,分别是表达式( Expression)、根对象( Root Object)、上下文环境( Context),下面分别讲解这三个要素,具体如下。
1、表达式
表达式是整个OGNL的核心,OGNL会根据表达式去对象中取值。所有OGNL操作都是针对表达式解析后进行的。它表明了此次OGNL操作要“做什么”。表达式就是一个带有语法含义的字符串,这个字符串规定了操作的类型和操作的内容。OGNL支持大量的表达式语法,不仅支持这种“链式”对象访问路径,还支持在表达式中进行简单的计算。
2、根对象(Root)
Root对象可以理解为OGNL的操作对象,表达式规定了“做什么”,而Root对象则规定了“对谁操作”。OGNL称为对象图导航语言,所谓对象图,即以任意一个对象为根,通过OGNL可以访问与这个对象关联的其它对象。
3、 Context对象
实际上OGNL的取值还需要一个上下文环境。设置了Root对象,OGNL可以对Root对象进行取值或写值等操作,Root对象所在环境就是OGNL的上下文环境( Context)。上下文环境规定了OGNL的操作“在哪里进行”。上下文环境 Context是一个Map类型的对象,在表达式中访问 Context中的对象,需要使用“#”号加上对象名称,即“#对象名称”的形式。
1.4 OGNL入门
下面通过一个示例来演示OGNL如何访问对象的方法:
My JSP 'ognl.jsp' starting page
在上面的测试中,我们调用了字符串的length方法,其中"'Kevin'.length( )"就是OGNL的表达式,最终输出结果为5。
二、值栈概述
2.1 什么是值栈
ValueStack是Struts的一个接口,字面意义为值栈, OgnlValueStack是ValueStack的实现类,客户端发起一个请求 struts2架构会创建一个action实例同时创建一个 OgnlValueStack 值栈实例,OgnlValueStack贯穿整个 Action的生命周期, struts2中使用OGNL将请求 Action的参数封装为对象存储到值栈中,并通过OGNL表达式读取值栈中的对象属性值。
2.2 值栈的内部结构
在OgnlValueStack中包括两部分:值栈和map。
Context:即 OgnlContext上下文,它是一个map结构,上下文中存储了一些引用, parameters、request、 session、 application等,上下文的Root为 Compoundroot。
Ognlcontext中的一些引用:
parameters:该Map中包含当前请求的请求参数;
request:该Map中包含当前 request对象中的所有属性 ;
session:该Map中包含当前 session对象中的所有属性 ;
application:该Map中包含当前 application对象中的所有属性;
attr:该Map按如下顺序来检索某个属性: request, session, application。
CompoundRoot:存储了 action实例,它作为 OgnlContext的Root对象
CompoundRoot继承 ArrayList实现压栈和出栈功能,拥有栈的特点,先进后出,后进先出,最后压进栈的数据在栈顶。我们把它称为对象栈。
struts2对原OGNL作出的改进就是Root使用 CompoundRooti(自定义栈),使用OgnlValueStack的findValue方法可以在 CompoundRoot中从栈顶向栈底找查找的对象的属性值。
CompoundRoot作为 OgnlContext的Root对象,并且在 CompoundRoot中action实例位于栈顶,当读取 action的属性值时会先从栈顶对象中找对应的属性,如果找不到则继续找栈中的其它对象,如果找到则停止查找。
2.3 获取值栈对象
【通过ActionContext获取值栈对象】:常用方式//第一种方式:获取值栈对象,
ActionContext context=ActionContext.getContext();
ValueStack stack=context.getValueStack();
【通过request域获取值栈对象】//第二种方式:获取值栈对象
ValueStack stack2=(ValueStack)ServletActionContext.getRequest().getAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY);
2.4 操作值栈
使用Action中的属性提供的get方法
调用值栈的push和set方法手动操作值栈//调用值栈对象的set方法
stack.set("setname", "Kevin");
//调用值栈对象的push方法
stack.push("Brank");
三、从值栈中获取数据
1. 字符串存取
【向值栈中存入字符串数据】
第一步:定义对象变量并生成变量的get方法//向值栈中存入字符串数据
private String username; public String getUsername(){ return username;
}
第二步:在执行方法中给变量赋值
public String execute() throws Exception { //给字符串复制
username="Kevin";
return SUCCESS;
}
【从值栈中获取字符串数据】
获取字符串
2. 对象存取
【向值栈中存入对象数据】
第一步:定义对象变量并声称get方法//向值栈中存入对象数据
private User user=new User(); public User getUser() { return user;
}
第二步:在执行方法中给对对象赋值
public String execute() throws Exception {
//给对象赋值
user.setAddress("China");
user.setPassword("admin");
user.setUsername("Kevin");
return SUCCESS;
}
【从值栈中获取对象数据】
获取对象
3. List集合存取
【向值栈中存入List集合】
第一步:定义集合变量并生成get方法//向值栈中存入列表数据
private List list=new ArrayList(); public List getList() { return list;
}
第二步:在执行方法中给list集合赋值
public String execute() throws Exception {
//在执行方法中给变量赋值
User user1=new User();
user1.setAddress("America");
user1.setPassword("admin");
user1.setUsername("Kevin");
User user2=new User();
user2.setAddress("British");
user2.setPassword("admin");
user2.setUsername("Brank");
list.add(user1);
list.add(user2);
return SUCCESS;
}
【从值栈中获取List集合】
获取列表第一种方式
获取列表第二种方式
获取列表第三种方式
使用foreach标签+EL表达式获取list方法设置的值
${user.username }
${user.password }
${user.address }