前言:关于前端请求参数传到后台,赋值的优先级取决于ognl表达式。
目录
一、OGNL概念
①OGNL的全称是Object Graph Navigation Language(对象图导航语言),它是一种强大的表达式语言
②OGNL相当于一个上下文(Context)概念,说白了上下文就是一个MAP结构,它实现了java.utils.Map 的接口。Struts框架默认就支持Ognl表达式语言。(从struts项目必须引入ognl.jar包可以看出)
③根对象只能有一个,而非根对象可以有多个, 非根对象要通过"#key"访问,根对象可以省略"#key"直接通过根对象属性就可以
二、使用
package com.zwf.test;
import ognl.Ognl;
import ognl.OgnlContext;
import ognl.OgnlException;
public class OnglExpression {
private OnglExpression() {
}
/**
* 根据OGNL表达式进行取值操作
*
* @param expression
* ognl表达式
* @param ctx
* ognl上下文
* @param rootObject
* ognl根对象
* @return
*/
public static Object getValue(String expression, OgnlContext ctx,
Object rootObject) {
try {
return Ognl.getValue(expression, ctx, rootObject);
} catch (OgnlException e) {
throw new RuntimeException(e);
}
}
/**
* 根据OGNL表达式进行赋值操作
*
* @param expression
* ognl表达式
* @param ctx
* ognl上下文
* @param rootObject
* ognl根对象
* @param value
* 值对象
*/
public static void setValue(String expression, OgnlContext ctx,
Object rootObject, Object value) {
try {
Ognl.setValue(expression, ctx, rootObject, value);
} catch (OgnlException e) {
throw new RuntimeException(e);
}
}
}
test.java
package com.zwf.test;
import ognl.OgnlContext;
import ognl.OgnlException;
public class Demo1 {
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);
}
}
打印结果:小李 张经理 小李 小明 孙经理 小芳
三、值栈特性
package com.zwf.test;
import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport;
import com.opensymphony.xwork2.util.ValueStack;
public class Demo7 extends ActionSupport{
/**
*
* 值栈的使用
*
*/
public String ognl1() {
// 栈:表示一个先进后出的数据结构
ValueStack vs = ActionContext.getContext().getValueStack();
// push方法把项压入栈顶
vs.push(new Employee("zs", 11));
vs.push(new Employee("ls", 11));
vs.push(new Employee("ww", 11));
// pop方法移除栈顶对象并作为此函数的值返回该对象
Employee e = (Employee) vs.pop();
System.out.println(e.getName());
e = (Employee) vs.pop();
System.out.println(e.getName());
e = (Employee) vs.pop();
System.out.println(e.getName());
return "bookEdit";
}
/**
* 此例用于模拟struts2的值栈计算过程
*
* @param args
*/
public String ognl2() {
ValueStack vs = ActionContext.getContext().getValueStack();
vs.push(new Employee("张员工", 1000));// 1
vs.push(new Student("小明", "s001"));// 0
System.out.println(vs.findValue("name"));
System.out.println(vs.findValue("salary"));
ActionContext ac = ActionContext.getContext();
return "bookEdit";
}
}
ognl1打印结果:ww ls zs
ognl2打印结果:小明 1000
四、验证疑问
通过值栈的特性可以推断出赋值的优先性,这要看我们的Struts2是如何压栈的。
通过上图可知,当一个参数name由前端请求发过来,它是先赋值到ModelDriver驱动类驱动的对
象中。如果该对象没有name属性,那么久赋值到TestAction类中的name属性。
这里只是大概介绍,讲解上篇的疑问!!!