OGNL表达式的介绍

OGNL的简介

OGNL是对象 - 图形导航语言(Object Graph Navigation Language)的缩写,它是一种功能强大的表达式语言,通过它简单一致的表达式语法,可以存取对象的任意属性,调用对象的方法,遍历整个对象的结构图,实现字段类型转化等功能。使用它的相同表达式去存取对象的属性。

OGNL与EL表达式的区别

用法区别:

OGNL是通常要结合Struts 2的标志一起使用,如<s:property value="#xx" /> struts页面中不能单独使用,el可以单独使用 ${sessionScope.username}

页面取值区别:
在这里插入图片描述
OgnlContext(ongl上下文)其实就是Map,
OgnlContext=根对象(1)+非根对象(N) (分为一个根对象和多个非根对象)
非根对象要通过"#key"访问,根对象可以省略"#key"
比如我们获取一个为根对象的的对象的属性,就只需要直接根据属性名就好了:String employeeName = (String) OnglExpression.getValue(“name”, ctx, e);
如果不是根对象: String managerName =(String)OnglExpression.getValue("#manager.name",ctx, e);

例:EL表达式是${book.name} OGNL是OgnlContext.get(#name)

重点:

1、一个上下文中只有一个根对象
2、取跟对象的值,只需要直接通过根对象属性即可
3、非根对象取值必须通过指定的上下文容器中的#key.属性去取。

OGNL案例

接下来我们看一个OGNL的根对象与非根对象的案例

package com.lst.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压栈

我们先来看一个案例

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";
	}

ValueStack是一个堆栈结构的容器 特点:先进后出
张雇员先进,小明同学后进。当要找name的时候,我们在Student中找到了name值,就不会再在Employee中取name值了,而找salary的时候,由于Student中没有salary,在Employee中找到了。所以就会显示2000.

后台定义属性和实现modelDriver接口,提供get和set方法

	private HttpServletRequest req;
	
	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;
	}
	
	/**
	 * implements ModelDriven
	 * @return
	 */
	public String accept1() {
		System.out.println("cal1:"+cal1);
		System.out.println("num1:"+num1);
		return "rs";
	}

前台向后台传值

<a href="${pageContext.request.contextPath }/sy/demo_accept1.action?num1=20&&num2=5">accept1</a>

当我们进行效果展示的时候,我们只能看到cal1中num1有值,而num1为null(空)。
我们已经提供了get方法,按道理来说,都是有值的,那为什么只有cal1中num1有值,而num1为null(空)?

这和ValueStack的压栈有关系,压栈的结构是先进后出,而ValueStack的取值是取到了值就返回,不会往下取。ValueStack对取值的压栈是先把xxxAction也就是我们通过set和get方法取值的方式先压入栈底,然后再把ModelDriver压入,所有会先取到通过ModelDriver来获取值的,而获取到了就返回了,导致后面的值不会获取了

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值