ognl原理

在上篇文章中,(struts2环境搭建和动态方法调用及传参)中,运用了3种方式接收来自jsp赋的值,那么这里有个疑问,当我在Action中,再封装一个num1,那么jsp传过来的值是会先赋值给ModelDriven里的还是Action中set/get中的,还是都赋值了?
首先第一步,在Action中set/get一个sum1,
然后随意写一个方法,在jsp页面中调用,通过控制台来查看是哪个赋了值,

public String accept1() {
		System.out.println("cal1:"+cal1);
		System.out.println("num1:"+num1);
		
		return "rs";
	}

输出结果为:
在这里插入图片描述
由此可得,接口中的赋了值,而set/get中的并没有赋值。

为什么会发生这种情况呢?经过专研,我了解到,这是因为他的底层运用的是OGNL表达式, 所以,接下来,给大家讲讲OGNL。

文章目录

了解OGNL

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

OGNL表达式中,是分为根对象,和非根对象,在一个ognlContext中只有一个根对象。非根对象是通过“#key”来调用,但根对象是可以省略的。

OGNL可以让我们用非常简单的表达式访问对象层,例如,当前环境的根对象为book,则表达式book.type[0].name可以访问到book的第一个type的name属性。

且,在运用ognlContext时,他也提供get/put方法,所以其实ognlContext是一个Map。
在这里插入图片描述
而到这里就有人会有疑问了,我并没有导OGNL的jar包,怎么这里有,
这是因为,当你通过仓库下载struts时,他的特性就是把有依赖的jar包也会一起下载,所以你可以在Libraries中看到这个jar包。
在这里插入图片描述

测试

接下来,通过案例来证实,OGNL的特性。

首先在一个class,写好set/getvalue方法。


	/**
	 * 根据OGNL表达式进行取值操作
	 * 
	 * @param expression
	 *            ognl表达式
	 * @param ctx
	 *            ognl上下文
	 * @param rootObject
	 *            ognl根对象
	 * @return
	 */
	public static Object getValue(String expression, OgnlContext ctx,
			Object rootObject) {
		try {
//			${book.type.name} 意味着你要在jsp上获取书籍的类别名称。
			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);
		}
	}

然后写一个demo来测试,

package com.zlk.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);
	}

}

输出如下
在这里插入图片描述
然后再通过一个demo来证实他的顺序。
在这里要注意的是实例化ValueStack有两种方式,
ValueStack vs = new OgnlValueStack();是老版本的实例。
而现在一般都是ValueStack vs = ServletActionContext.getContext().getValueStack();来获得。

package com.zlk.web;

import org.apache.struts2.ServletActionContext;

import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.util.ValueStack;
import com.zlk.test.Employee;
import com.zlk.test.Student;

public class DemoAction {
	/**
	 * 此例用于模拟struts2的值栈计算过程
	 * 
	 * @param args
	 */
	public String test1() {
		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"));
		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";
	}
}


运行main方法后,test1方法输出是
在这里插入图片描述

test2方法输出的是:
在这里插入图片描述
两个方法都说明,他是从上往下取值的,特点就是遇到的第一个属性进行输出。也就是先进后出。

所以回到最初的问题,
通过ModelDriven接口的获得了值,set/get缺没赋值,这就是因为
ognl首先会将set/get等先放进去,然后再放入接口的。
所以值第一次进来的时候,找到的是接口中的num1。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值