`
airlink
  • 浏览: 4572 次
  • 性别: Icon_minigender_1
  • 来自: 北京
文章分类
社区版块
存档分类
最新评论

用测试理解如何使用ognl: 读取和设置包含筛选条件的列表元素的属性。

阅读更多

先转一篇介绍

OGNL 语言介绍与实践 

 

官网上的资料比较精炼,例子很少,还是写一些测试代码来验证用法:

 

包装一下:

package org.airlinkmatrix.test.ognl;

import ognl.Ognl;
import ognl.OgnlContext;
import ognl.OgnlException;

public class OgnlExpression {
	
	private static final OgnlContext DEFAULT_CONTEXT = (OgnlContext)Ognl.createDefaultContext(null);
	
	private Object expression;
	private OgnlContext context;

	public void setContext(OgnlContext context) {
		this.context = context;
	}

	public OgnlContext getContext() {
		if (context==null){
			context = DEFAULT_CONTEXT;
		}
		return context;
	}
	
	public OgnlExpression(String expressionString) throws OgnlException
	{
		super();
		expression = Ognl.parseExpression(expressionString);
	}

	public Object getExpression()
	{
		return expression;
	}

	public Object getValue(Object rootObject) throws OgnlException
	{
		return Ognl.getValue(getExpression(), getContext(), rootObject);
	}

	public void setValue(Object rootObject, Object value) throws OgnlException
	{
		Ognl.setValue(getExpression(), getContext(), rootObject, value);
	}

}

 测试类:

package org.airlinkmatrix.test.ognl;

import java.util.ArrayList;
import java.util.List;
import java.util.Vector;

import org.airlinkmatrix.test.ognl.OgnlExpression;

import junit.framework.TestCase;
import ognl.OgnlException;

public class OgnlExpressionTest extends TestCase {

	class Root{
		private Object[] array;
		public Object[] getArray() {
			return array;
		}
		public void setArray(Object[] array) {
			this.array = array;
		}
		
		private Vector vector;
		public Vector getVector() {
			return vector;
		}
		public void setVector(Vector vector) {
			this.vector = vector;
		}
		public Root() {
			vector = new Vector();
			vector.add(new Part("Michael","002"));
			vector.add(new Part("Alex","001"));
			vector.add(new Part("Joseph","003"));
			vector.add(new Part("Alex","004"));
			
			array = new Part[4];
			array[0] = new Part("Michael","002");
			array[1] = new Part("Alex","001");
			array[2] = new Part("Joseph","003");
			array[3] = new Part("Alex","004");
		}
	}
	
	class Part{
		private String id;
		public String getId() {
			return id;
		}

		public void setId(String id) {
			this.id = id;
		}

		private String name;

		public Part(String name,String id) {
			super();
			this.name = name;
			this.id = id;
		}

		public String getName() {
			return name;
		}

		public void setName(String name) {
			this.name = name;
		}

		public String toString() {
//			return "name='"+name+"',id='"+id+"'";
			return name;
		}

		public boolean equals(Object obj) {
			if (!(obj instanceof Part)){
				return false;
			}
			Part anotherPart = (Part)obj;
			if (this.name==null||this.id==null){
				return false;
			}
			if (!this.name.equals(anotherPart.getName())){
				return false;
			}
			if (!this.id.equals(anotherPart.getId())){
				return false;
			}
			return true;
		}
		
	}
	
	private Root ROOT;

	protected void setUp() throws Exception {
		super.setUp();
		ROOT = new Root();
	}

	protected void tearDown() throws Exception {
		super.tearDown();
	}
	
	public void testGetValue_ROOT_vector() {
		try {
			OgnlExpression expression = null;
			ArrayList valueList = null;
			
			//?号表示匹配所有记录,
			expression = new OgnlExpression("vector.{? #this.name == 'Alex' }");
			valueList = (ArrayList)expression.getValue(ROOT);
			assertEquals(new Part("Alex","001"),valueList.get(0));
			assertEquals(new Part("Alex","004"),valueList.get(1));
			
			//^号表示匹配第一条记录
			expression = new OgnlExpression("vector.{^ #this.name == 'Alex' }");
			valueList = (ArrayList)expression.getValue(ROOT);
			assertEquals(new Part("Alex","001"),valueList.get(0));

			//$号表示匹配最后一条记录
			expression = new OgnlExpression("vector.{$ #this.name == 'Alex' }");
			valueList = (ArrayList)expression.getValue(ROOT);
			assertEquals(new Part("Alex","004"),valueList.get(0));

			//?号匹配多条记录,所以有可能选到第二条记录
			expression = new OgnlExpression("vector.{? #this.name == 'Alex' }[1].id");
			assertEquals("004",expression.getValue(ROOT));

			//可以接着取记录属性
			expression = new OgnlExpression("vector.{^ #this.name == 'Alex' }[0].id");
			assertEquals("001",expression.getValue(ROOT));

			try{
				//^号匹配单条记录,第二条记录一定是空的,会报IndexOutOfBoundsException
				expression = new OgnlExpression("vector.{^ #this.name == 'Alex' }[1].id");
				expression.getValue(ROOT);
				fail("expect IndexOutOfBoundsException");
			}catch (IndexOutOfBoundsException ex){
				
			}catch (Exception otherEx){
				fail("expect IndexOutOfBoundsException");
			}
			
			try{
				//^号匹配单条记录,如果匹配不到数据,仍然会报IndexOutOfBoundsException
				//这个与官网的文档是有出入的,^号并不能避免IndexOutOfBoundsException
				expression = new OgnlExpression("vector.{^ #this.name == 'Wrong Name' }[0].id");
				expression.getValue(ROOT);
				fail("expect IndexOutOfBoundsException");
			}catch (IndexOutOfBoundsException ex){
				
			}catch (Exception otherEx){
				fail("expect IndexOutOfBoundsException");
			}
			
		} catch (OgnlException e) {
			e.printStackTrace();
			fail("oops!");
		}
	}

	
	public void testGetValue_ROOT_array() {
		
		try {
			OgnlExpression expression = null;
			ArrayList valueList = null;
			
			//?号表示匹配所有记录,
			expression = new OgnlExpression("array.{? #this.name == 'Alex' }");
			valueList = (ArrayList)expression.getValue(ROOT);
			assertEquals(new Part("Alex","001"),valueList.get(0));
			assertEquals(new Part("Alex","004"),valueList.get(1));
			
			//^号表示匹配第一条记录
			expression = new OgnlExpression("array.{^ #this.name == 'Alex' }");
			valueList = (ArrayList)expression.getValue(ROOT);
			assertEquals(new Part("Alex","001"),valueList.get(0));

			//$号表示匹配最后一条记录
			expression = new OgnlExpression("array.{$ #this.name == 'Alex' }");
			valueList = (ArrayList)expression.getValue(ROOT);
			assertEquals(new Part("Alex","004"),valueList.get(0));

			//?号匹配多条记录,所以有可能选到第二条记录
			expression = new OgnlExpression("array.{? #this.name == 'Alex' }[1].id");
			assertEquals("004",expression.getValue(ROOT));

			//可以接着取记录属性
			expression = new OgnlExpression("array.{^ #this.name == 'Alex' }[0].id");
			assertEquals("001",expression.getValue(ROOT));

			try{
				//^号匹配单条记录,第二条记录一定是空的,会报IndexOutOfBoundsException
				expression = new OgnlExpression("array.{^ #this.name == 'Alex' }[1].id");
				expression.getValue(ROOT);
				fail("expect IndexOutOfBoundsException");
			}catch (IndexOutOfBoundsException ex){
				
			}catch (Exception otherEx){
				fail("expect IndexOutOfBoundsException");
			}
			
			try{
				//^号匹配单条记录,如果匹配不到数据,仍然会报IndexOutOfBoundsException
				//这个与官网的文档是有出入的,^号并不能避免IndexOutOfBoundsException
				expression = new OgnlExpression("array.{^ #this.name == 'Wrong Name' }[0].id");
				expression.getValue(ROOT);
				fail("expect IndexOutOfBoundsException");
			}catch (IndexOutOfBoundsException ex){
				
			}catch (Exception otherEx){
				fail("expect IndexOutOfBoundsException");
			}
			
		} catch (OgnlException e) {
			e.printStackTrace();
			fail("oops!");
		}
	}
	
	public void testSetValue_ROOT_Normal() {
		
		try {
			OgnlExpression expression = null;
			
			expression = new OgnlExpression("array.{^ #this.name == 'Alex' }[0].id");
			expression.setValue(ROOT,"007");
			
			assertEquals("007",((Part)ROOT.getArray()[1]).getId());
			
		} catch (OgnlException e) {
			e.printStackTrace();
			fail("oops!");
		}
	}
	
	
}
 

读取、设置对象图内部的一个值没有多大问题了。但,ognl 组件是否能够配置为设置列表中每一个元素的某字段为同一值呢?

例如,对象图如下:

ROOT

   Part

     id="1"

   Part

     id="2"

   Part

     id="3"

 

其中ROOT拥有Part的一个列表(java.util.List)

 

有没有一种写法,能利用ognl表达式一次将上述对象的值改为:

ROOT

   Part

     id="007"

   Part

     id= "007"

   Part

     id= "007"

 

在ognl语言参考中没有找到。

尝试自己扩展一个吧。

分享到:
评论
3 楼 airlink 2009-11-21  
这样就支持了出现单个列表[]符号的多记录赋值。

下一步可以考虑支持多个列表[]符号的两重或多重列表记录的赋值。

2 楼 airlink 2009-11-21  
<p>继续,为了保留原来的类和测试功能作为参考,我新写了一个ExtendedOgnlExpression包装类和相应的测试ExtendedOgnlExpressionTest</p>
<p> </p>
<p>ExtendedOgnlExpressionTest如下:</p>
<p> </p>
<pre name="code" class="java">package org.airlinkmatrix.test.ognl;

import java.util.Vector;

import junit.framework.TestCase;
import ognl.OgnlException;

public class ExtendedOgnlExpressionTest extends TestCase {

class Root{
private Object[] array;
public Object[] getArray() {
return array;
}
public void setArray(Object[] array) {
this.array = array;
}

private Vector vector;
public Vector getVector() {
return vector;
}
public void setVector(Vector vector) {
this.vector = vector;
}
public Root() {
vector = new Vector();
vector.add(new Part("Michael","002"));
vector.add(new Part("Alex","001"));
vector.add(new Part("Joseph","003"));
vector.add(new Part("Alex","004"));

array = new Part[4];
array[0] = new Part("Michael","002");
array[1] = new Part("Alex","001");
array[2] = new Part("Joseph","003");
array[3] = new Part("Alex","004");
}
}

class Part{
private String id;
public String getId() {
return id;
}

public void setId(String id) {
this.id = id;
}

private String name;

public Part(String name,String id) {
super();
this.name = name;
this.id = id;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public String toString() {
// return "name='"+name+"',id='"+id+"'";
return name;
}

public boolean equals(Object obj) {
if (!(obj instanceof Part)){
return false;
}
Part anotherPart = (Part)obj;
if (this.name==null||this.id==null){
return false;
}
if (!this.name.equals(anotherPart.getName())){
return false;
}
if (!this.id.equals(anotherPart.getId())){
return false;
}
return true;
}

}

private Root ROOT;

protected void setUp() throws Exception {
super.setUp();
ROOT = new Root();
}

protected void tearDown() throws Exception {
super.tearDown();
}

public void testSetValue_ROOT_Normal() {

try {
OgnlExpression expression = null;

expression = new OgnlExpression("array.{^ #this.name == 'Alex' }[0].id");
expression.setValue(ROOT,"007");

assertEquals("007",((Part)ROOT.getArray()[1]).getId());

} catch (OgnlException e) {
e.printStackTrace();
fail("oops!");
}
}
/**
* 扩展OgnlExpression setValue语法,例如
* "array.{^ #this.name == 'Alex' }[].id"
* 表示将设置列表中name等于Alex的所有元素的id属性
*/
public void testSetValue_ROOT_Multiple_Normal() {

try {
ExtendedOgnlExpression expression = null;

expression = new ExtendedOgnlExpression("array.{? #this.name == 'Alex' }[].id");
expression.setValue(ROOT, "007");

//匹配的第二第四条记录值改变了
assertEquals("007",((Part)ROOT.getArray()[1]).getId());
assertEquals("007",((Part)ROOT.getArray()[3]).getId());

//没匹配的第一第三条记录值没变
assertEquals("002",((Part)ROOT.getArray()[0]).getId());
assertEquals("003",((Part)ROOT.getArray()[2]).getId());
} catch (OgnlException e) {
e.printStackTrace();
fail("oops!");
}
}
}
</pre>
<p> 简单起见,做了一些copy&amp;paste。 修正测试用例的一个小错误, ^  改为 ? ,即</p>
<pre name="code" class="java">new ExtendedOgnlExpression("array.{? #this.name == 'Alex' }[].id");</pre>
<p> </p>
<p>实现思路是在[]中填充index,然后依赖IndexOutOfBoundsException来判断结果是否已经遍历结束(有点dirty,但貌似没其他办法)。</p>
<p> </p>
<p>实现如下:</p>
<pre name="code" class="java">package org.airlinkmatrix.test.ognl;

import ognl.Ognl;
import ognl.OgnlContext;
import ognl.OgnlException;

public class ExtendedOgnlExpression {

private static final OgnlContext DEFAULT_CONTEXT = (OgnlContext)Ognl.createDefaultContext(null);

private OgnlContext context;

private String expressionString;

public void setContext(OgnlContext context) {
this.context = context;
}

public OgnlContext getContext() {
if (context==null){
context = DEFAULT_CONTEXT;
}
return context;
}

public ExtendedOgnlExpression(String expressionString)
{
super();
this.expressionString = expressionString;
}

public Object getExpression(String expressionString) throws OgnlException
{
return Ognl.parseExpression(expressionString);
}

public Object getValue(Object rootObject) throws OgnlException
{
if (expressionString.indexOf("[]")&gt;0){
throw new OgnlException("getValue do not support []");
}
return Ognl.getValue(getExpression(expressionString), getContext(), rootObject);
}

public void setValue(Object rootObject, Object value) throws OgnlException
{
if (expressionString.indexOf("[]")&gt;0){
setMultipleValue(rootObject, value);

}else{
setValue(rootObject, value, expressionString);
}

}

private void setMultipleValue(Object rootObject, Object value)
throws OgnlException {
//依次在[]中填入0,1,2,...,设置目标值,直到超出匹配列表的最后一个,发生IndexOutOfBoundsException为止
int i = 0;
while(true){
try {
setValue(rootObject, value, addIndex(expressionString,i));
} catch (IndexOutOfBoundsException e) {
e.printStackTrace();
break;
}
i++;
}
}

private void setValue(Object rootObject, Object value,String expressionString) throws OgnlException {
Ognl.setValue(getExpression(expressionString), getContext(), rootObject, value);
}

private String addIndex(String expressionString, int i) {
int index = expressionString.indexOf("[]");
String str = expressionString.substring(0,index+1)
+String.valueOf(i)
+expressionString.substring(index+1);

return str;
}

}
</pre>
<p> </p>
<p>运行测试,成功:)</p>
<p> </p>
<p> </p>
<p> </p>
1 楼 airlink 2009-11-21  
<p>写一个测试当用例:</p>
<p> </p>
<pre name="code" class="java"> /**
* 扩展OgnlExpression setValue语法,例如
* "array.{^ #this.name == 'Alex' }[].id"
* 表示将设置列表中name等于Alex的所有元素的id属性
*/
public void testSetValue_ROOT_Multiple_Normal() {

try {
OgnlExpression expression = null;

expression = new OgnlExpression("array.{^ #this.name == 'Alex' }[].id");
expression.setValue(ROOT, "007");

//匹配的第二第四条记录值改变了
assertEquals("007",((Part)ROOT.getArray()[1]).getId());
assertEquals("007",((Part)ROOT.getArray()[3]).getId());

//匹配的第一第三条记录值没变
assertEquals("002",((Part)ROOT.getArray()[0]).getId());
assertEquals("003",((Part)ROOT.getArray()[2]).getId());
} catch (OgnlException e) {
e.printStackTrace();
fail("oops!");
}
}</pre>
 
<p> </p>
<p> </p>

相关推荐

    ognl-3.2.21-API文档-中文版.zip

    赠送jar包:ognl-3.2.21.jar; 赠送原API文档:ognl-3.2.21-javadoc.jar; 赠送源代码:ognl-3.2.21-sources.jar; ...人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。

    ognl-3.1.12-API文档-中文版.zip

    赠送jar包:ognl-3.1.12.jar; 赠送原API文档:ognl-3.1.12-javadoc.jar; 赠送源代码:ognl-3.1.12-sources.jar; ...人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。

    使用的ognl的chm

    **OGNL(Object-Graph Navigation Language)**是一种强大的表达式语言,主要用于获取和设置Java对象的属性。在Java开发中,尤其是Web应用框架如Struts2和Hibernate中,OGNL被广泛使用。它允许开发者以简洁的方式...

    OGNL学习笔记,包含struts2中ognl的各种用法

    在示例代码中,`&lt;s:property&gt;`标签使用OGNL表达式`#request.req`、`#session.sess`和`#application.app`分别获取了Request、Session和Application范围内的属性值。这展示了OGNL如何灵活地访问不同级别的数据。 其次...

    OGNL表达式的使用及文档

    1. **表达式语法**:OGNL表达式通常由对象名、点号和属性名组成,例如`user.name`,表示获取`user`对象的`name`属性。也可以通过索引访问集合元素,如`list[0]`。 2. **上下文(Context)**:在OGNL中,上下文包含...

    ognl-3.2.21-API文档-中英对照版.zip

    赠送jar包:ognl-3.2.21.jar; 赠送原API文档:ognl-3.2.21-javadoc.jar; 赠送源代码:ognl-3.2.21-...人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。 双语对照,边学技术、边学英语。

    struts2-OGNL表达式测试

    在“struts2-OGNL表达式测试”中,你可能会看到各种OGNL表达式的使用示例,比如访问和修改对象属性、动态调用方法、处理集合和执行条件判断等。通过这些测试,你可以更好地理解和掌握OGNL在实际开发中的运用,确保在...

    ognl-3.1.12-API文档-中英对照版.zip

    赠送jar包:ognl-3.1.12.jar; 赠送原API文档:ognl-3.1.12-javadoc.jar; 赠送源代码:ognl-3.1.12-...人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。 双语对照,边学技术、边学英语。

    ognl项目文件和源代码

    OGNL (Object-Graph Navigation Language) 是一个强大的表达式语言,主要用于在Java应用程序中获取和设置对象属性。它在Java社区中被广泛使用,尤其是在Struts2框架中,作为视图和模型之间的数据绑定工具。这个...

    JS:ognl表达式

    OGNL (Object-Graph Navigation Language) 是一种强大的表达式语言,用于获取和设置 Java 对象的属性。它最初是由 John Rose 和 Luke Blanshard 开发,并被广泛应用于多种框架中,尤其是在 Spring Expression ...

    ognl源码和ognl帮助文档.rar

    在这个压缩包中,包含的是OGNL的源码和帮助文档,这对于深入理解OGNL的工作原理以及在Struts2中的应用非常有帮助。 OGNL的核心特性包括: 1. **对象导航**:OGNL允许我们通过`.`操作符来访问对象的属性,甚至可以...

    OGNL访问OGNL上下文和Action上下文Struts2rarfdsafsdfasdfas

    OGNL上下文(OGNL Context)和Action上下文是Struts2中两个重要的概念,它们对于理解和使用OGNL表达式至关重要。 **OGNL上下文(OGNL Context)** OGNL上下文是OGNL表达式执行时的工作空间,它存储了键值对,其中键...

    struts2中ognl表达式的理解和用法

    在Struts2中,OGNL(Object-Graph Navigation Language)是一种强大的表达式语言,用于获取和设置对象的属性。这篇博客将深入探讨OGNL表达式的理解和用法。 OGNL的全称是对象图导航语言,它的主要功能是在对象模型...

    Ognl参考手册pdf

    - **OGNL对象索引属性**:使用OGNL提供的特殊方式来访问对象属性。 4. **方法调用**:可以调用对象的方法,如`object.method()`。 5. **变量引用**:表达式中可以通过变量名引用变量值。 6. **括号表达式**:使用...

    ognl-2.6.11.jar 官方源码

    1. **阅读源码**:通过查看源码,了解OGNL解析器如何将表达式转化为操作,以及如何处理不同类型的对象和属性。 2. **实验和调试**:创建示例项目,实践使用OGNL表达式,观察其行为,同时通过调试源码来理解其工作...

    OGNL语言中文资料

    OGNL,全称为Object-Graph Navigation Language,是一种强大的表达式语言,主要被用于获取和设置Java对象的属性。它在Java开发中广泛应用于Spring框架、Struts2等,提供了简洁而灵活的方式来访问和操作对象的属性。...

    Struts2 使用OGNL表达式

    10. **OGNL与拦截器**:Struts2的拦截器机制允许在执行Action之前或之后对数据进行处理,而OGNL则可以用来设置或读取这些数据。 总的来说,了解并熟练掌握OGNL是深入理解Struts2框架的关键。通过使用OGNL,开发者...

    ognl.zip 版本3.0.6

    `PropertyAccessor`是OGNL框架中的核心组件,负责处理对象属性的读取和写入操作。当出现这个错误时,意味着你的项目中没有包含OGNL库或者其依赖项不完整,需要确保将ognl.jar添加到项目的类路径中。 **OGNL的用途**...

    ognl 源码: struts2 所用的 ognl 包源码

    OGNL (Object-Graph Navigation Language) 是一种强大的表达式语言,用于获取和设置 Java 对象的属性。在 Struts2 框架中,OGNL 作为默认的表示层语言,用于视图与模型之间的数据绑定。Struts2 的核心功能之一就是...

    ognl标签的使用案例

    OGNL (Object-Graph Navigation Language) 是一种强大的表达式语言,主要用于在Java应用程序中获取和设置对象属性。在Web开发领域,尤其是与Struts2框架结合时,OGNL标签被广泛用于视图层,允许开发者以简洁的方式...

Global site tag (gtag.js) - Google Analytics