BeanUtils.copyProperties使用總結(jié)以及注意事項(xiàng)說明
1.前言
開發(fā)過程中,講一個(gè)對(duì)象的屬性和值賦值到另一個(gè)對(duì)象上,大量使用了get、set方法,看著很臃腫,思考下肯定不只有我有這種想法,所以技術(shù)上肯定有方法能解決這個(gè)問題,所以查閱了一些資料發(fā)現(xiàn)了BeanUtils.copyProperties這個(gè)方法以下是這次所有的總結(jié)以及使用時(shí)的注意事項(xiàng)。
使用org.springframework.beans.BeanUtils.copyProperties方法進(jìn)行對(duì)象之間屬性的賦值,避免通過get、set方法一個(gè)一個(gè)屬性的賦值。
2.一般使用
BeanUtils是這個(gè)包里比較常用的一個(gè)工具類,該方法定義如下:
public static void copyProperties(java.lang.Object dest,java.lang.Object orig) throws java.lang.IllegalAccessException, java.lang.reflect.InvocationTargetException
如 果你有兩個(gè)具有很多相同屬性的JavaBean,一個(gè)很常見的情況就是Struts里的PO對(duì)象(持久對(duì)象)和對(duì)應(yīng)的ActionForm,例如 Teacher和TeacherForm。我們一般會(huì)在Action里從ActionForm構(gòu)造一個(gè)PO對(duì)象,傳統(tǒng)的方式是使用類似下面的語句對(duì)屬性逐 個(gè)賦值:
//1得到TeacherForm TeacherForm teacherForm=(TeacherForm)form; //2構(gòu)造Teacher對(duì)象 Teacher teacher=new Teacher(); //3賦值 teacher.setName(teacherForm.getName()); teacher.setAge(teacherForm.getAge()); teacher.setGender(teacherForm.getGender()); teacher.setMajor(teacherForm.getMajor()); teacher.setDepartment(teacherForm.getDepartment()); //4持久化Teacher對(duì)象到數(shù)據(jù)庫 HibernateDAO=; HibernateDAO.save(teacher);
而使用BeanUtils后,代碼就大大改觀了,如下所示:
//1得到TeacherForm TeacherForm teacherForm=(TeacherForm)form; //2構(gòu)造Teacher對(duì)象 Teacher teacher=new Teacher(); //3賦值 BeanUtils.copyProperties(teacher,teacherForm); //4持久化Teacher對(duì)象到數(shù)據(jù)庫 HibernateDAO=; HibernateDAO.save(teacher);
如 果Teacher和TeacherForm間存在名稱不相同的屬性,則BeanUtils不對(duì)這些屬性進(jìn)行處理,需要程序員手動(dòng)處理。
例如 Teacher包含modifyDate(該屬性記錄最后修改日期,不需要用戶在界面中輸入)屬性而TeacherForm無此屬性,那么在上面代碼的 copyProperties()后還要加上一句:
teacher.setModifyDate(new Date());
3.拷貝屬性時(shí)忽略空值
使用BeanUtils.copyProperties有一個(gè)問題就是當(dāng)src對(duì)象的鍵值為Null時(shí)
就會(huì)把target對(duì)象的對(duì)應(yīng)鍵值覆蓋成空了,這明顯不是我們想要的,以下這個(gè)方法可以解決
/** * 復(fù)制屬性,過濾掉不復(fù)制的屬性 */ public static void copyBeanProperties( final Object source,//1,待復(fù)制的原始對(duì)象 final Object target,//2,復(fù)制后的結(jié)果對(duì)象 //3,獲取保存你不需要復(fù)制的屬性名 final Collection<String> excludes = new ArrayList<String>(); final PropertyDescriptor[] propertyDescriptors = BeanUtils.getPropertyDescriptors(source.getClass()); for(final PropertyDescriptor propertyDescriptor : propertyDescriptors){ String propName = propertyDescriptor.getName(); if(!includes.contains(propName)){ excludes.add(propName); } } //4,復(fù)制操作 BeanUtils.copyProperties(source, target, excludes.toArray(new String[excludes.size()])); }
使用案例
public static String[] getNullPropertyNames (Object source) { final BeanWrapper src = new BeanWrapperImpl(source); java.beans.PropertyDescriptor[] pds = src.getPropertyDescriptors(); Set<String> emptyNames = new HashSet<String>(); for(java.beans.PropertyDescriptor pd : pds) { Object srcValue = src.getPropertyValue(pd.getName()); if (srcValue == null) emptyNames.add(pd.getName()); } String[] result = new String[emptyNames.size()]; return emptyNames.toArray(result); } public static void copyPropertiesIgnoreNull(Object src, Object target){ BeanUtils.copyProperties(src, target, getNullPropertyNames(src)); }
4.使用注意事項(xiàng)(1)
Property.copyProperties()和BeanUtils.copyProperties()
除BeanUtils外還有一個(gè)名為PropertyUtils的工具類,它也提供copyProperties()方法
- 1.無論是org.springframework.beans或者org.apache.commons.beanutils,與get/set方式相比,都存在性能問題。
- 2.效率由高到底:get/set 》PropertyUtils 》BeanUtils。
- 3.PropertyUtils和BeanUtils兩個(gè)工具類都是對(duì)bean之間存在屬性名相同的屬性進(jìn)行處理,無論是源bean或者是目標(biāo)bean中多出來的屬性均不處理。
- 4.具體來說:BeanUtils.copyProperties()可以在一定范圍內(nèi)進(jìn)行類型轉(zhuǎn)換,同時(shí)還要注意一些不能轉(zhuǎn)換時(shí)候,會(huì)將默認(rèn)null值轉(zhuǎn)化成0;Property.copyProperties()則是嚴(yán)格的類型轉(zhuǎn)化,必須類型和屬性名完全一致才轉(zhuǎn)化。對(duì)于null的處理:PropertyUtils支持為null的場(chǎng)景;BeanUtils對(duì)部分屬性不支持null,具體如下:
- a. java.util.Date類型不支持,但是它的自雷java.sql.Date是被支持的。java.util.Date直接copy會(huì)報(bào)異常;
- b. Boolean,Integer,Long等不支持,會(huì)將null轉(zhuǎn)化為0;
- c. String支持,轉(zhuǎn)化后依然為null。
- 5.BeanUtils的高級(jí)功能org.apache.commons.beanutils.Converter接口可以自定義類型轉(zhuǎn)化,也可以對(duì)部分類型數(shù)據(jù)的null值進(jìn)行特殊處理,如ConvertUtils.register(new DateConverter(null), java.util.Date.class);但是PropertyUtils沒有。
另外:值得注意的是,在測(cè)試過程中發(fā)現(xiàn),commons-beanutils-1.8.0.jar版本中的BeanUtils類,支持Byte到Integer或int的轉(zhuǎn)化。說明實(shí)際使用過程中,我們還是要多看源碼,多做測(cè)試,并且注意版本號(hào)升級(jí)帶來的微小變化。
BeanUtils支持的轉(zhuǎn)換類型如下:
* 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
這里要注意一點(diǎn),java.util.Date是不被支持的,而它的子類java.sql.Date是被支持的。因此如果對(duì)象包含時(shí)間類型的屬性,且希望被轉(zhuǎn)換的時(shí)候,一定要使用java.sql.Date類型。否則在轉(zhuǎn)換時(shí)會(huì)提示argument mistype異常。
5.使用注意事項(xiàng)(2)
在Java中copyProperties() 這個(gè)方法出處有兩個(gè)地方,
BeanUtils是org.springframework.beans.BeanUtils
BeanUtils是org.apache.commons.beanutils.BeanUtils
下面具體說說他們的用法和區(qū)別。這個(gè)方法在不同的包下面,而這兩個(gè)類的copyProperties()方法里面?zhèn)鬟f的參數(shù)賦值是相反的。
例如:a,b為對(duì)象 BeanUtils.copyProperties(a, b);
BeanUtils是org.springframework.beans.BeanUtils //a拷貝到b //a1 源文件,b1 目標(biāo)文件 public static void copyProperties(Object a1, Object b1) throws BeansException { copyProperties(a1, b1, null, (String[])null); }
BeanUtils是org.apache.commons.beanutils.BeanUtils
//b拷貝到a //a2目標(biāo)文件 ,b2 原始的,源文件 public static void copyProperties(Object a2, Object b2) throws IllegalAccessException, InvocationTargetException { BeanUtilsBean.getInstance().copyProperties(a2, b2); }
引用包出處不一樣,意思就不一樣,使用的時(shí)候一定要看清楚是哪個(gè)包下面的。
6.使用注意事項(xiàng)(3)
方法使用起來是不是即省事又舒服?but。。。get/set寫代碼省事是要付出代價(jià)的,那就是使用BeanUtils的運(yùn)行成本也驚人!BeanUtils所花費(fèi)的時(shí)間要超過取數(shù) 據(jù)、將其復(fù)制到對(duì)應(yīng)的 value對(duì)象(通過手動(dòng)調(diào)用get和set方法),以及通過串行化將其返回到遠(yuǎn)程的客戶機(jī)的時(shí)間總和。所以要小心使用這種威力!有失必有得,反之亦然。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java實(shí)現(xiàn)的時(shí)間戳與date對(duì)象相互轉(zhuǎn)換功能示例
這篇文章主要介紹了Java實(shí)現(xiàn)的時(shí)間戳與date對(duì)象相互轉(zhuǎn)換功能,結(jié)合具體實(shí)例形式分析了java日期與時(shí)間戳類型的表示與轉(zhuǎn)換相關(guān)操作技巧,需要的朋友可以參考下2017-06-06Java設(shè)計(jì)模式之單例模式實(shí)例詳解【懶漢式與餓漢式】
這篇文章主要介紹了Java設(shè)計(jì)模式之單例模式,簡(jiǎn)單說明了單例模式的原理并結(jié)合具體實(shí)例形式分析了單例模式中懶漢式與餓漢式的具體實(shí)現(xiàn)與使用技巧,需要的朋友可以參考下2017-09-09dockerfile-maven-plugin極簡(jiǎn)教程(推薦)
這篇文章主要介紹了dockerfile-maven-plugin極簡(jiǎn)教程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10Spring的兩種事務(wù)管理機(jī)制的基本概念和demo示例
Spring事務(wù)包括聲明式事務(wù)管理和注解式事務(wù)管理,我們通過概念和小demo的形式一步一步地來一起學(xué)習(xí)這個(gè)知識(shí)點(diǎn),需要的朋友可以參考下2023-07-07SpringBoot整合thymeleaf 報(bào)錯(cuò)的解決方案
這篇文章主要介紹了SpringBoot整合thymeleaf 報(bào)錯(cuò)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08idea中導(dǎo)入項(xiàng)目后main方法無法Run的解決
這篇文章主要介紹了idea中導(dǎo)入項(xiàng)目后main方法無法Run的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。2023-03-03由ArrayList來深入理解Java中的fail-fast機(jī)制
fail-fast俗稱快速失敗,是在多線程進(jìn)行迭代操作時(shí)產(chǎn)生沖突的一種異常拋出機(jī)制,下面我們就由ArrayList來深入理解Java中的fail-fast機(jī)制.2016-05-05Java并發(fā)編程之關(guān)鍵字volatile的深入解析
提高java的并發(fā)編程,就不得不提volatile關(guān)鍵字,不管是在面試還是實(shí)際開發(fā)中volatile都是一個(gè)應(yīng)該掌握的技能,這篇文章主要給大家介紹了關(guān)于Java并發(fā)編程之關(guān)鍵字volatile的相關(guān)資料,需要的朋友可以參考下2021-09-09