ognl简介
OGNL是Object-Graph Navigation Language的缩写,它是一种功能强大的表达式语言,通过它简单一致的表达式语法,可以存取对象的任意属性,调用对象的方法,遍历整个对象的结构图,实现字段类型转化等功能。它使用相同的表达式去存取对象的属性。这样可以更好的取得数据。
EL 和OGNL之间的区别
EL表达式获取一个对象的属性是${book.name}
OGNL获取则是OgnlContext.get(#name)
实体类:
HelloAction
public class HelloAction implements ModelDriven<Cal>,ServletRequestAware {
private HttpServletRequest req;
private Cal cal1= new Cal();
private Cal cal2;//需要get set方法
private String sex;//需要get set方法
private String num1;//需要get set方法 再昨天的基础上再加一个属性
}
Cal
public class Cal {
private String num1;
private String num2;
提出疑问获取的值应该是那个:
1:是两个同时获取?
2:获取上面一个?
3:获取下面一个?
public String accept1() {
System.out.println("Cal1:"+cal1);
System.out.println("num1:"+num1);
jsp页面的代码:
<a href="${pageContext.request.contextPath }/sy/demo_accept1.action?num1=20&&num2=5">accept1</a>
后台结构展示为:
所以说再这个里面set/get方式是失效了的
那我们今天就来看一下 struts 的传值优先级
在这里我们先了解一下OLNL表达式
根对象和非根对象的理解(区别):
1.一个上下文中只有一个根对象
2.取根对象的值,只需要直接通过根对象属性即可
3.非根对象取值必须通过指定的上下文容器中的#key.属性去取。
在这里我们来举个例子
OgnlContext(ongl上下文)其实就是Map (教室、老师、学生)
map 相当于教室
OgnlContext=根对象(1)+非根对象(N)
老师:根对象 1(根对象只有一个 )
学生:非根对象 n(非根对象可以含有多个)
接下来:我们来导入几个类
OgnlException类
import ognl.Ognl;
import ognl.OgnlContext;
import ognl.OgnlException;
/**
* 用于OGNL表达计算的一个工具类
*
*/
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) {
// $(book.c.name) 意味着你要在jsp上获取书籍的类别的名称 ctx.get(#{name})
// ctx.get($(Book.c.name)) 找到具体的位置
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);
}
}
}
接下来我们就来测试一下Ognl取值和赋值
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);//根对象为小芳
}
}
运行结果为:
我们之前导入的类是会报错的
需要改为
ValueStack vs = ServletActionContext.getContext().getValueStack();
//ognl表达式只存在struts里面 不然会报空指针异常
// 需要请求 再main方法里面没有值
public class DemoAction {
//ognl表达式只存在struts里面 不然会报空指针异常
// 需要请求 再main方法里面没有值
/**
* 此例用于模拟struts2的值栈计算过程
*
* @param args
*/
public String test1() {
//ValueStack 是一个堆栈的容器 特点先进后出
ValueStack vs = ServletActionContext.getContext().getValueStack();
vs.push(new Employee("张雇员", 2000));// 1
vs.push(new Student("小明同学", "s001"));// 0
System.out.println(vs.findValue("name"));//小明同学
System.out.println(vs.findValue("salary"));//2000
return "rs";
}
/**
*
* 值栈的使用
*
*/
public String test2() {
// 栈:表示一个先进后出的数据结构
ValueStack vs = ServletActionContext.getContext().getValueStack();
// push方法把项压入栈顶
vs.push(new Employee("zs", 22));
vs.push(new Employee("ls", 22));
vs.push(new Employee("ww", 22));
// 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 "rs";
}
}
在struts.sy.xml里面配置
<action name="/stack_*" class="com.xyx.test.DemoAction" method="{1}">
<result name="rs">/rs.jsp</result>
</action>
jsp页面
<h3>调用OGNL</h3>
<a href="${pageContext.request.contextPath }/sy/stack_test1.action">ognl1</a>
结果:
在这里我们用的是堆栈
遵循 先进后出 的原则