BeanUtils & PropertyUtils & MethodUtils類使用方法
一、簡介:
BeanUtils提供對 Java反射和自省API的包裝。其主要目的是利用反射機制對JavaBean的屬性進行處理。我們知道,一個JavaBean通常包含了大量的屬性,很多情況下,對JavaBean的處理導致大量get/set代碼堆積,增加了代碼長度和閱讀代碼的難度。
二、用法:
BeanUtils是這個包里比較常用的一個工具類,這里只介紹它的copyProperties()方法。該方法定義如下:
throws java.lang.IllegalAccessException,
java.lang.reflect.InvocationTargetException
如果你有兩個具有很多相同屬性的JavaBean,一個很常見的情況就是Struts里的PO對象(持久對象)和對應的ActionForm,例如 Teacher和TeacherForm。我們一般會在Action里從ActionForm構造一個PO對象,傳統的方式是使用類似下面的語句對屬性逐個賦值:
TeacherForm teacherForm=(TeacherForm)form;
//構造Teacher對象
Teacher teacher=new Teacher();
//賦值
teacher.setName(teacherForm.getName());
teacher.setAge(teacherForm.getAge());
teacher.setGender(teacherForm.getGender());
teacher.setMajor(teacherForm.getMajor());
teacher.setDepartment(teacherForm.getDepartment());
//持久化Teacher對象到數據庫
HibernateDAO=;
HibernateDAO.save(teacher);
而使用BeanUtils后,代碼就大大改觀了,如下所示:
//得到TeacherForm
TeacherForm teacherForm=(TeacherForm)form;
//構造Teacher對象
Teacher teacher=new Teacher();
//賦值
BeanUtils.copyProperties(teacher,teacherForm);
//持久化Teacher對象到數據庫
HibernateDAO=;
HibernateDAO.save(teacher);
如果Teacher和TeacherForm間存在名稱不相同的屬性,則BeanUtils不對這些屬性進行處理,需要程序員手動處理。例如 Teacher包含modifyDate(該屬性記錄最后修改日期,不需要用戶在界面中輸入)屬性而TeacherForm無此屬性,那么在上面代碼的 copyProperties()后還要加上一句:
怎么樣,很方便吧!除BeanUtils外還有一個名為PropertyUtils的工具類,它也提供copyProperties()方法,作用與 BeanUtils的同名方法十分相似,主要的區別在于后者提供類型轉換功能,即發現兩個JavaBean的同名屬性為不同類型時,在支持的數據類型范圍內進行轉換,而前者不支持這個功能,但是速度會更快一些。BeanUtils支持的轉換類型如下:
- java.lang.BigDecimal
- java.lang.BigInteger
- boolean and java.lang.Boolean
- byte and java.lang.Byte
- char and java.lang.Character
- java.lang.Class
- double and java.lang.Double
- float and java.lang.Float
- int and java.lang.Integer
- long and java.lang.Long
- short and java.lang.Short
- java.lang.String
- java.sql.Date
- java.sql.Time
- java.sql.Timestamp
這里要注意一點,java.util.Date是不被支持的,而它的子類java.sql.Date是被支持的。因此如果對象包含時間類型的屬性,且希望被轉換的時候,一定要使用java.sql.Date類型。否則在轉換時會提示argument mistype異常。
三、優缺點:
Apache Jakarta Commons項目非常有用。我曾在許多不同的項目上或直接或間接地使用各種流行的commons組件。其中的一個強大的組件就是BeanUtils。我將說明如何使用BeanUtils將local實體bean轉換為對應的value 對象:
上面的代碼從aLocal對象復制屬性到aValue對象。它相當簡單!它不管local(或對應的value)對象有多少個屬性,只管進行復制。我們假設local對象有100個屬性。上面的代碼使我們可以無需鍵入至少100行的冗長、容易出錯和反復的get和set方法調用。這太棒了!太強大了!太有用了!
BeanUtils.copyProperties 與 PropertyUtils.copyProperties 都是拷貝對象屬性的方法,BeanUtils 支持類型轉換,而 PropertyUtils 不支持。但是 BeanUtils 不允許對象的屬性值為 null,PropertyUtils 可以拷貝屬性值 null 的對象。 如果對象屬性值為 null,BeanUtils.copyProperties 方法會報 commons.beanutils.ConversionException: No value specified 錯誤。
現在,還有一個壞消息:使用BeanUtils的成本驚人地昂貴!我做了一個簡單的測試,BeanUtils所花費的時間要超過取數據、將其復制到對應的value對象(通過手動調用get和set方法),以及通過串行化將其返回到遠程的客戶機的時間總和。所以要小心使用這種威力!
MethodUtils類使用方法:
import java.util.Map;
import org.apache.commons.beanutils.MethodUtils;
import org.apache.commons.beanutils.PropertyUtils;
public class TestPropertyUtils {
public static void main(String[] args) throws Exception{
Entity entity = new Entity();
//通過PropertyUtils的getProperty方法獲取指定屬性的值
Integer id = (Integer)PropertyUtils.getProperty(entity, "id");
String name = (String)PropertyUtils.getProperty(entity, "name");
System.out.println("id = " + id + " name = " + name);
//調用PropertyUtils的setProperty方法設置entity的指定屬性
PropertyUtils.setProperty(entity, "name", "心夢帆影");
System.out.println("name = " + entity.getName());
//通過PropertyUtils的describe方法把entity的所有屬性與屬性值封裝進Map中
Map map = PropertyUtils.describe(entity);
System.out.println("id = " + map.get("id") + " name = " + map.get("name"));
//通過MethodUtils的invokeMethod方法,執行指定的entity中的方法(無參的情況)
System.out.println( MethodUtils.invokeMethod(entity, "haha", null) );
//通過MethodUtils的invokeMethod方法,執行指定的entity中的方法(1參的情況)
MethodUtils.invokeMethod(entity, "sayHelle", "心夢帆影");
//通過MethodUtils的invokeMethod方法,執行指定的entity中的方法(多參的情況)
Object[] params = new Object[]{new Integer(10),new Integer(12)};
String msg = (String)MethodUtils.invokeMethod(entity, "countAges", params);
System.out.println(msg);
}
}