BeanUtils & PropertyUtils & MethodUtils類使用方法

作者: gundumw100  來源: JavaEye  發布時間: 2010-09-04 10:12  閱讀: 2883 次  推薦: 0   原文鏈接   [收藏]  

一、簡介: 
  BeanUtils提供對 Java反射和自省API的包裝。其主要目的是利用反射機制對JavaBean的屬性進行處理。我們知道,一個JavaBean通常包含了大量的屬性,很多情況下,對JavaBean的處理導致大量get/set代碼堆積,增加了代碼長度和閱讀代碼的難度。 
二、用法: 
  BeanUtils是這個包里比較常用的一個工具類,這里只介紹它的copyProperties()方法。該方法定義如下: 

 
public static void copyProperties(java.lang.Object dest,java.lang.Object orig)
throws java.lang.IllegalAccessException,
java.lang.reflect.InvocationTargetException

  如果你有兩個具有很多相同屬性的JavaBean,一個很常見的情況就是Struts里的PO對象(持久對象)和對應的ActionForm,例如 Teacher和TeacherForm。我們一般會在Action里從ActionForm構造一個PO對象,傳統的方式是使用類似下面的語句對屬性逐個賦值: 

 

 
//得到TeacherForm
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()后還要加上一句: 

 
teacher.setModifyDate(new Date());

怎么樣,很方便吧!除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 對象: 

 
BeanUtils.copyProperties(aValue, aLocal);

上面的代碼從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類使用方法: 

 
package rong.propertyUtils;
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);
}
}
0
0
 
 
 

文章列表

arrow
arrow
    全站熱搜
    創作者介紹
    創作者 大師兄 的頭像
    大師兄

    IT工程師數位筆記本

    大師兄 發表在 痞客邦 留言(0) 人氣()