首页 > 其他分享 >OGNL表达式语言介绍

OGNL表达式语言介绍

时间:2023-04-26 13:31:32浏览次数:39  
标签:username valueStack 语言 OgnlValueStack OGNL model public 表达式 users


OGNL表达式语言介绍

[code]
OGNL介绍
OGNL是Object-Graph Navigation Language的缩写,它是一种功能强大的表达式语言(Expression Language,简称为EL),通过它简单一致的表达式语法,可以存取对象的任意属性,调用对象的方法,遍历整个对象的结构图,实现字段类型转化等功能。它使用相同的表达式去存取对象的属性。
XWork遵循“不要重复地发明同一个轮子”的理论,它的表达式语言核心用的就是这个OGNL。我们先来看看一个简单的例子:
还记得我们用户注册的那个例子吗?我们输入框的name用到的名字就是OGNL的表达式,比如:用户名的输入框:“<input type="text" name="user.username">”,在用户注册成功之后我们要显示用户注册的信息,用了“<ww:property value="user.username"/>”。Input输入框里的“user.username”,它解析成Java语句为:getUser().setUsername();,property标签里的“user.username”解析为Java语句:getUser.getUsername();。
我们的两个表达式都是相同的,但前一个保存对象属性的值,后一个是取得对象属性的值。表达式语言简单、易懂却又功能强大,关于OGNL更多的介绍可以去http://www.ognl.org,那里有很详细的文档

值堆栈-OgnlValueStack
OGNL在框架中的应用,最主要是支持我们的值堆栈(Value Stack)——OgnlValueStack,它主要的功能是通过表达式语言来存取对象的属性。用户界面输入数据,它会根据保存表达式将数据依次保存到它堆栈的对象中,业务操作完成,结果数据会通过表达式被获取、输出。
还记得我们用户注册的例子吗?下面我们用一段程序来演示它向 OgnlValueStack中保存、取得数据的步骤:

// DemoRegisterValueStack 

package example.register; 

import com.opensymphony.xwork.util.OgnlValueStack; 


public class DemoRegisterValueStack { 

 public void demo(){ 

 RegisterAction action = new RegisterAction(); 

 OgnlValueStack valueStack= new OgnlValueStack(); 

 valueStack.push(action); 


 valueStack.setValue("user.username","Babydavic"); 

 System.out.println("username = "+valueStack.findValue("user.username")); 

 } 


 public static void main(String[] args) { 

DemoRegisterValueStack demoValueStack = new DemoRegisterValueStack(); 

 demoValueStack.demo(); 

 } 

}


我们来看一看它的demo()方法:
1、 创建我们的Action(RegisterAction)类的对象action,将action对象压入堆栈valueStack中。在WebWrok中 Action的创建、入栈是在DefaultActionInvocation构造函数中进行的,详细介绍见:ServletDispatcher原理。
2、 通过表达式语言,调用堆栈对象的get()、set()方法,设置该对象的值。
public void setValue(String expr, Object value)
语句:valueStack.setValue("user.username","Babydavic");
的作用等同于:action.getUser().setUsername("Babydavic");
3、 通过表达式语言,去堆栈对象中查找我们前面保存的值,并在控制台打印。valueStack.findValue("user.username")等同与语句:
action.getUser().getUsername()
最后控制台打印的结果:
username = Babydavic


CompoundRoot
在OgnlValueStack中,一个堆栈其实是一个List。查看OgnlValueStack你会发现,堆栈就是 com.opensymphony.xwork.util.CompoundRoot类的对象:

public class CompoundRoot extends ArrayList { 

 //~ Constructors / 

 public CompoundRoot() { 

} 


 public CompoundRoot(List list) { 

 super(list); 

} 


 //~ Methods 

 public CompoundRoot cutStack(int index) { 

 return new CompoundRoot(subList(index, size())); 

} 


 public Object peek() { 

 return get(0); 

 } 

 public Object pop() { 

 return remove(0); 

 } 

 public void push(Object o) { 

 add(0, o); 

 } 

}



我们通过表达式向堆栈对象操作时,我们并不知道堆栈中有哪些对象。OgnlValueStack会根据堆栈由上向下的顺序(先入栈在下面,最后入栈在最上面)依次去查找与表达式匹配的对象方法,找到即进行相应的存取操作。假设后面对象也有相同的方法,将不会被调用。
下面我们看一个对 OgnlValueStack操作的程序,它主要演示了如何对Map对象的存取和OgnlValueStack堆栈的原理

package example.register; 


import com.opensymphony.xwork.util.OgnlValueStack; 


public class DemoGroupValueStack { 


 public void demoAction(){ 

 DemoGroupAction action = new DemoGroupAction(); 

 OgnlValueStack valueStack= new OgnlValueStack(); 

 valueStack.push(action); 


 User zhao = new User(); 

 zhao.setUsername("zhao"); 

 zhao.setEmail("[email protected]"); 


 User qian = new User(); 

 qian.setUsername("qian"); 

 qian.setEmail("[email protected]"); 


 valueStack.setValue("users['zhao']",zhao); 

 valueStack.setValue("users['qian']",qian); 



 System.out.println("users['zhao'] = "+valueStack.findValue("users['zhao']")); 

 System.out.println("users['qian'] = "+valueStack.findValue("users['qian']")); 

 System.out.println("users size = "+valueStack.findValue("users.size")); 


 System.out.println("allUserName[0] = "+valueStack.findValue("allUserName[0]")); 

 } 


 public void demoModels(){ 


 User model_a = new User(); 

 model_a.setUsername("model_a"); 

 User model_b = new User(); 

 model_b.setUsername("model_b"); 

 User model_c = new User(); 

 model_c.setUsername("model_c"); 


 OgnlValueStack valueStack= new OgnlValueStack(); 

 valueStack.push(model_a); 

 valueStack.push(model_b); 

 valueStack.push(model_c); 


 System.out.println("username = "+valueStack.findValue("username")); 

 System.out.println("[1].username = "+valueStack.findValue("[1].username")); 

 System.out.println("[0].toString = "+valueStack.findValue("[0]")); 

 System.out.println("[1].toString = "+valueStack.findValue("[1]")); 

 System.out.println("[2].toString = "+valueStack.findValue("[2]")); 


 } 

 public static void main(String[] args) { 

 DemoGroupValueStack demoValueStack = new DemoGroupValueStack(); 

 demoValueStack.demoAction(); 

 demoValueStack.demoModels(); 

 } 

} 



package example.register; 


import java.util.ArrayList; 

import java.util.HashMap; 

import java.util.List; 

import java.util.Map; 


public class DemoGroupAction { 


 private Map users = new HashMap(); 


 public Map getUsers(){ 

 return this.users; 

 } 


 public List getAllUserName(){ 

 return new ArrayList(users.keySet()); 

 } 

 public String execute(){ 

 //执行业务操作 

 return null; 

 } 

 public String toString(){ 

 return users.toString(); 

 } 

}


注意:1、 Map属性的存取,它的表达式语言如:users['zhao'],注意它用’’来引用HashMap的key字符串。
2、 demoModels()方法演示了OgnlValueStack中堆栈的原理,请特别注意它的 [0].toString、[1].toString、[2].toString,它们依次调用堆栈中对象的toString()方法,并逐一的减少堆栈最上面的对象。
控制台输出的结果如下:
users['zhao'] = username=zhao;password=null;[email protected];age=0
users['qian'] = username=qian;password=null;[email protected];age=0
users size = 2
allUserName[0] = qian

username = model_c
[1].username = model_b
[0].toString = [username=model_c;password=null;email=null;age=0, username=model_b;password=null;email=null;age=0, username=model_a;password=null;email=null;age=0]
[1].toString = [username=model_b;password=null;email=null;age=0, username=model_a;password=null;email=null;age=0]
[2].toString = [username=model_a;password=null;email=null;age=0]
[/code]


标签:username,valueStack,语言,OgnlValueStack,OGNL,model,public,表达式,users
From: https://blog.51cto.com/u_16087012/6227129

相关文章

  • 精通struts2的ognl表达式
    精通struts2的ognl表达式[code]Struts2中内置了OGNL表达式的支持,使得Struts2的具有比Struts1更为强大的数据访问的功能。本文主要讲解OGNL的使用方法,并不会去讲解一些原理性的东西,想要了解的朋友可以自己去查阅相关的资料在OGNL的使用主要分以下部分来......
  • 12 | C语言中的函数类型和函数指针类型
    函数类型和函数指针类型的区别,让我们先看一个例子#include<iostream>usingnamespacestd;typedefint(Func)(int);typedefint(*Func_p)(int);intf(inta){returna;}intmain(){Func_pp=f;Func*p_ptr=f;cout<<p(0)<<endl;cout<<p_ptr(1)......
  • pid算法函数实现,c语言版
     #include<stdio.h>floatpid(floatsetpoint,floatprocess_variable,floatkp,floatki,floatkd,floatdt,float*integral,float*last_error){//Calculateerrorfloaterror=setpoint-process_variable;//Calculateintegral......
  • C语言基础知识
    一维数组inta[2]={1,2},一维数组名a代表的是数组第一个元素的地址,不代表数组中所有元素。二维数组inta[3][4]总共是12个元素,可以当作3行4列来看待,这十二个元素的名字依次是:a[0][0],a[0][1],a[0][2],a[0][3]a[1][0],a[1][1],a[1][2],a[1][3]a[2][0],a[2][1],a[2][2],a[2][3]......
  • R语言中aggregate 函数
     001、测试数据框studentID<-seq(1,20)gender<-rep(c("M","M","F","F","F"),4)math<-rep(c(92,86,85,74,82),4)english<-rep(c(76,69,82,71,80),4)class<-rep(c(paste0(c(1,2,......
  • 左值、左值表达式、左值引用 C++
     本文topics什么是左值和右值什么是左值表达式、右值表达式引用的分类之前学习Java的时候只记得在赋值表达式左边的就是左值,右边的就是右值。这个说法没有错,但今天又在C++中有学习到了关于左右值得概念,在此梳理了一些这些知识点。左值和右值左值:左值是一个对象或变量,可以代表......
  • 不同语言加载不同字号,设置到资源文件中,进行引用
    在资源文件夹创建一个类在App.xaml文件中引用这个类的空间命名,并把这个类添加到资源在页面中应用在使用其他语言时,开启新的子线程依然会使用区域语言.net4.5后使用可以一次性解决varculture=newCultureInfo("en-US");    CultureInfo.DefaultThreadCurrent......
  • Go语言入门12(协程 goroutine)
    协程进程和线程进程​ 当运行一个应用程序的时候,操作系统会为这个应用程序启动一个进程。可以将这个进程看作一个包含了应用程序在运行中需要用到和维护的各种资源的容器。这些资源包括但不限于内存地址空间、文件和设备的句柄以及线程线程​ 一个线程是一个执行空间,这个空间......
  • R语言宏观经济学:IS-LM曲线可视化货币市场均衡
    全文链接:http://tecdat.cn/?p=32249原文出处:拓端数据部落公众号凯恩斯相关理论主要是美国20世纪30年代的经济危机而提出的,主张政府干预经济,实行宏观调控。按照希克斯的观点,灵活偏好(L)和货币数量(M)决定着货币市场的均衡,而人们持有的货币数量既决定于利率(i),又决定于收入(y)的......
  • 正则表达式(补
    1.限定符a*:表示a重复0次及以上a+:表示a重复1次及以上a?:表示a重复0次或1次a{n}:表示a重复n次a{n,}:表示a重复n次或更多次a{n,m}:表示a重复n到m次注:如果需要判断字符串的重复:(字符串)限定符2.或运算符"|"eg.a(cat|dog):匹配的字符串是acat或adog3.字符类我们想匹配由a、b、c......