简介:
OGNL(Object-Graph Navigation Language)的全称是对象图导航语言,它是一种功能强大的开源表达式语言,比EL(只能从域或内置对象中)表达式更强大,使用这种表达式语言,可以通过某种表达式语法,OGNL可以存取Java任意对象的任意属性,调用Java对象的方法,同时能够自动实现必要的类型转换。如果把表达式看作是一个带有语义的字符串,那么OGNL无疑成为了这个语义字符串与Java对象之间沟通的桥梁。
OGNL的操作实际上就是围绕着OGNL结构的三个要素而进行的,分别是表达式(Expresssion)、根对象(Root Object)、上下文环境(Context)
1. 表达式
表达式是整个 OGNL 的核心,OGNL 会根据表达式到对象中取值。所有 OGNL 操作都是针对表达式解析后进行的,它表明了此次 OGNL 操作要“做什么”。
实际上,表达式就是一个带有语法含义的字符串,这个字符串规定了操作的类型和操作的内容。
2. 根对象
根对象可以理解为 OGNL 的操作对象,OGNL 可以对根对象进行取值或写值等操作,表达式规定了“做什么”,而根对象则规定了“对谁操作”。实际上根对象所在的环境就是 OGNL 的上下文对象环境。
3. 上下文对象
上下文对象规定了 OGNL 操作“在哪里进行”。context 对象是一个 Map 类型的对象,在表达式中访问 context 中的对象,需要使用 # 号加对象名称,即“# 对象名称”的形式
OgnlContext(ongl上下文)其实就是Map ,分根对象和丰根对象
非根对象要通过"#key"访问,根对象可以省略"#key"
1、一个上下文中只有一个根对象
2、取跟对象的值,只需要直接通过根对象属性即可
3、非根对象取值必须通过指定的上下文容器中的#key.属性去取。
OGNL取值的一个案例
package com.chenjiahao.test;
import ognl.OgnlContext;
import ognl.OgnlException;
public class Demo1 {
/**
* @param args
* @throws OgnlException
*/
public static void main(String[] args) {
//一个叫小李的员工
Employee e = new Employee();
e.setName("小李");
//张经理的管理人员
Manager m = new Manager();
m.setName("张经理");
// 创建OGNL下文,而OGNL上下文实际上就是一个Map对象
OgnlContext ctx = new OgnlContext();
// 将员工和经理放到OGNL上下文当中去
ctx.put("employee", e);
ctx.put("manager", m);
//一个公司有很多老板,只有一个员工
ctx.setRoot(e);// 设置OGNL上下文的根对象
/** ********************** 取值操作 *************************** */
// 表达式name将执行e.getName(),因为e对象是根对象(请注意根对象和非根对象表达式的区别)
String employeeName = (String) OnglExpression.getValue("name", ctx, e);
System.out.println(employeeName);//小李
// 表达式#manager.name将执行m.getName(),注意:如果访问的不是根对象那么必须在前面加上一个名称空间,例如:#manager.name
String managerName = (String) OnglExpression.getValue("#manager.name",
ctx, e);
System.out.println(managerName);//张经理
// 当然根对象也可以使用#employee.name表达式进行访问
employeeName = (String) OnglExpression.getValue("#employee.name", ctx,
e);
System.out.println(employeeName);//小李
/** ********************** 赋值操作 *************************** */
OnglExpression.setValue("name", ctx, e, "小明");
employeeName = (String) OnglExpression.getValue("name", ctx, e);
System.out.println(employeeName);
OnglExpression.setValue("#manager.name", ctx, e, "孙经理");
managerName = (String) OnglExpression.getValue("#manager.name", ctx, e);
System.out.println(managerName);
OnglExpression.setValue("#employee.name", ctx, e, "小芳");
employeeName = (String) OnglExpression.getValue("name", ctx, e);
System.out.println(employeeName);
}
}
结果:
OGNL向ValueStack压栈
前台向后台传值
<a href="${pageContext.request.contextPath}/sy/demo_accept1.action?num1=20&&num2=5">accept1</a>
后台代码:定义属性实现modelDriven接口,提供get/set方法
public class HelloAction implements ModelDriven<Cal>{
private HttpServletRequest request;
private Cal cal1=new Cal();
private Cal cal2;
private String sex;
private String num1;
public String getNum1() {
return num1;
}
public void setNum1(String num1) {
this.num1 = num1;
}
public Cal getCal2() {
return cal2;
}
public void setCal2(Cal cal2) {
this.cal2 = cal2;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public String accept1() {
System.out.println("cal1:"+cal1);
System.out.println("num1:"+num1);
return "rs";
}
}
解释为什么cal1.num1有值,而num1取不到值?
我们知道Struts后台三种取值方式是 实现modelDrivern 接口,提供get/set方法,还有一种是 类实例.属性名 ;上面已经提供get/set方法和实现modelDrivern 按道理是可以获取到值的,然而这里取不到。
原因: ValueStack压栈,我们知道压栈是先进后出 ;ValueStack取到值就不会往下去取值了。