PropertyUtils.getProperty和PropertyUtils.setProperty的用法详解
背景
一般情况下,在Java中你可以通过get方法轻松获取beans中的属性值。但是,当你事先不知道beans的类型或者将要访问或修改的属性名时,该怎么办?Java语言中提供了一些像java.beans.Introspector
这 样类,实现了在运行时检测Java类并确定属性get和set方法的名称,结合Java中的反射机制就可以调用这些方法了。然而,这些APIs使用起来比 较困难,并且将Java类中一些不必要的底层结构暴露给了开发人员。BeanUtils包中的APIs试图简化动态获取和设置bean属性的过程。
BeanUtils包中的PropertyUtils类中的一些静态方法实现了上面的功能,稍后会详细介绍。首先,介绍一些有用的定义:
JavaBean支持的属性类型一般可以划分成三类--标准的JavaBeans规范支持其中的一些,也有一部分只有BeanUtils包支持:
- Simple(单值) -- 单值或量,有个一可以访问或修改的属性。值的类型可能是Java语言的原生类型(如:int型),简单的类(如:java.lang.String),或 者一个复杂类的对象,这个类可能来自Java语言或者来自应用程序再或者来自应用程序中的一个类库。
- Indexed(索 引) -- 索引的属性,属性中存放有序对象(都是同类型的)的集合,每个对象都可以通过一个非负的整数值(或下标)来获取。另外,所有值的集合可以使用一个数组来设 置或者获取。作为一个JavaBeans规范的扩展,BeanUtils包认为所有底层数据类型为java.util.List(或List的一个实现) 的属性都可以被索引。
- Mapped(映射) -- 作为一个标准JavaBeans APIs的扩展, BeanUtils包认为所有底层数据类型为java.util.Map的属性都可以被"映射"。你可以通过String类型的key值来设置或者获取对应的值。
PropertyUtils类中提供了各种API方法用来获取和设置上述三种类型的属性。在下面的程序片段中,假设在bean类中都定义了如下的方法:
- public class Employee {
- public Address getAddress(String type);
- public void setAddress(String type, Address address);
- public Employee getSubordinate(int index);
- public void setSubordinate(int index, Employee subordinate);
- public String getFirstName();
- public void setFirstName(String firstName);
- public String getLastName();
- public void setLastName(String lastName);
访问基本属性
获取和设置simple属性很简单。在Javadocs中查看下面两个方法:
- PropertyUtils.getSimpleProperty(Object bean, String name)
- PropertyUtils.setSimpleProperty(Object bean, String name, Object value)
使用这两个方法,你可以动态地修改employee的name属性:
- Employee employee = ...;
- String firstName = (String) PropertyUtils.getSimpleProperty(employee, "firstName");
- String lastName = (String) PropertyUtils.getSimpleProperty(employee, "lastName");
- ... manipulate the values ...
- PropertyUtils.setSimpleProperty(employee, "firstName", firstName);
- PropertyUtils.setSimpleProperty(employee, "lastName", lastName);
对于indexed(索引)属性,你有两种选择 - 你既可以在属性名后面添加方括号在里面放上一个下标,也可以在调用方法时将其作为一个独立参数:
- PropertyUtils.getIndexedProperty(Object bean, String name)
- PropertyUtils.getIndexedProperty(Object bean, String name, int index)
- PropertyUtils.setIndexedProperty(Object bean, String name, Object value)
- PropertyUtils.setIndexedProperty(Object bean, String name, int index, Object value)
属性名的下标只能是整数常量。如果你想获取的项的索引是计算出来的,你可以将属性名和索引作为字符串组合起来。例如,你可以向下面这样做:
- Employee employee = ...;
- int index = ...;
- String name = "subordinate[" + index + "]";
- Employee subordinate = (Employee) PropertyUtils.getIndexedProperty(employee, name);
- Employee employee = ...;
- int index = ...;
- Employee subordinate = (Employee) PropertyUtils.getIndexedProperty(employee, "subordinate", index);
类似的,获取和设置mapped(映射)属性的方法同样有两对。与indexed(索引)不同的是额外的属性是用括号括起来的(“(”和“)”)而不是方括号,并且获取和设置值时如同从底层的map中获取和设置值一样。
- PropertyUtils.getMappedProperty(Object bean, String name)
- PropertyUtils.getMappedProperty(Object bean, String name, String key)
- PropertyUtils.setMappedProperty(Object bean, String name, Object value)
- PropertyUtils.setMappedProperty(Object bean, String name, String key, Object value)
例如,你可以使用下面两种方法设置employee的家庭住址:
- Employee employee = ...;
- Address address = ...;
- PropertyUtils.setMappedProperty(employee, "address(home)", address);
- Employee employee = ...;
- Address address = ...;
- PropertyUtils.setMappedProperty(employee, "address", "home", address);
访问嵌套属性
在上面的例子中,我们假设你将bean作为第一个参数传入PropertyUtils方法,并希望获取指定属性的值。然而,如果属性的值是一个Java对象,并且你希望进一步获取这个Java对象的某个属性的值?
例如,假设你事实上想要获取的值是employee家庭住址中的city属性。使用标准的Java编程技术直接获取bean的对应属性,你可以这样写:
- String city = employee.getAddress("home").getCity();
使用PropertyUtils类中的等效机制被称为嵌套属性访问。使用这种方法,你将访问路径上的属性的名称用“.”拼接起来 -- 与你在JavaScript执行嵌套属性访问的方式非常相似。
- PropertyUtils.getNestedProperty(Object bean, String name)
- PropertyUtils.setNestedProperty(Object bean, String name, Object value)
PropertyUtils中等效于上面的Java代码将是这样:
- String city = (String) PropertyUtils.getNestedProperty(employee, "address(home).city");
最后,方便起见,PropertyUtils提供了如下一组方法,它们接收simple、indexed和mapped属性的任意组合方法,支持任意层次的嵌套:
- PropertyUtils.getProperty(Object bean, String name)
- PropertyUtils.setProperty(Object bean, String name, Object value)
你可以像这样使用:
- Employee employee = ...;
- String city = (String) PropertyUtils.getProperty(employee, "subordinate[3].address(home).city");
原文链接:http://commons.apache.org/beanutils/api/org/apache/commons/beanutils/package-summary.html#standard