简介:
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取到值就不会往下去取值了。