国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術文章
文章詳情頁

淺析Java中Apache BeanUtils和Spring BeanUtils的用法

瀏覽:3日期:2023-08-02 15:14:11

# 前言

在我們實際項目開發過程中,我們經常需要將不同的兩個對象實例進行屬性復制,從而基于源對象的屬性信息進行后續操作,而不改變源對象的屬性信息,比如DTO數據傳輸對象和數據對象DO,我們需要將DO對象進行屬性復制到DTO,但是對象格式又不一樣,所以我們需要編寫映射代碼將對象中的屬性值從一種類型轉換成另一種類型。

# 對象拷貝

在具體介紹兩種 BeanUtils 之前,先來補充一些基礎知識。它們兩種工具本質上就是對象拷貝工具,而對象拷貝又分為深拷貝和淺拷貝,下面進行詳細解釋。

# 什么是淺拷貝和深拷貝

在Java中,除了 基本數據類型之外,還存在 類的實例對象這個引用數據類型,而一般使用 “=”號做賦值操作的時候,對于基本數據類型,實際上是拷貝的它的值,但是對于對象而言,其實賦值的只是這個對象的引用,將原對象的引用傳遞過去,他們實際還是指向的同一個對象。

而淺拷貝和深拷貝就是在這個基礎上做的區分,如果在拷貝這個對象的時候,只對基本數據類型進行了拷貝,而對引用數據類型只是進行引用的傳遞,而沒有真實的創建一個新的對象,則認為是淺拷貝。反之,在對引用數據類型進行拷貝的時候,創建了一個新的對象,并且復制其內的成員變量,則認為是深拷貝。

簡單來說:

淺拷貝:對基本數據類型進行值傳遞,對引用數據類型進行引用傳遞般的拷貝,此為淺拷貝

深拷貝:對基本數據類型進行值傳遞,對引用數據類型,創建一個新的對象,并復制其內容,此為深拷貝。

淺析Java中Apache BeanUtils和Spring BeanUtils的用法

# BeanUtils

前面簡單講了一下對象拷貝的一些知識,下面就來具體看下兩種 BeanUtils 工具

# Apache 的 BeanUtils

首先來看一個非常簡單的BeanUtils的例子

publicclass PersonSource { private Integer id; private String username; private String password; private Integer age; // getters/setters omiited}publicclass PersonDest { private Integer id; private String username; private Integer age; // getters/setters omiited}publicclass TestApacheBeanUtils { public static void main(String[] args) throws InvocationTargetException, IllegalAccessException { //下面只是用于單獨測試 PersonSource personSource = new PersonSource(1, 'pjmike', '12345', 21); PersonDest personDest = new PersonDest(); BeanUtils.copyProperties(personDest,personSource); System.out.println('persondest: '+personDest); }}persondest: PersonDest{id=1, username=’pjmike’, age=21}

從上面的例子可以看出,對象拷貝非常簡單,BeanUtils最常用的方法就是:

//將源對象中的值拷貝到目標對象//將源對象中的值拷貝到目標對象public static void copyProperties(Object dest, Object orig) throws IllegalAccessException, InvocationTargetException { BeanUtilsBean.getInstance().copyProperties(dest, orig);}

但是由于 Apache下的BeanUtils對象拷貝性能太差,不建議使用,而且在阿里巴巴Java開發規約插件上也明確指出:

Ali-Check | 避免用Apache Beanutils進行屬性的copy。

commons-beantutils 對于對象拷貝加了很多的檢驗,包括類型的轉換,甚至還會檢驗對象所屬的類的可訪問性,可謂相當復雜,這也造就了它的差勁的性能,具體實現代碼如下:

public void copyProperties(final Object dest, final Object orig) throws IllegalAccessException, InvocationTargetException { // Validate existence of the specified beans if (dest == null) { thrownew IllegalArgumentException ('No destination bean specified'); } if (orig == null) { thrownew IllegalArgumentException('No origin bean specified'); } if (log.isDebugEnabled()) { log.debug('BeanUtils.copyProperties(' + dest + ', ' + orig + ')'); } // Copy the properties, converting as necessary if (orig instanceof DynaBean) { final DynaProperty[] origDescriptors = ((DynaBean) orig).getDynaClass().getDynaProperties(); for (DynaProperty origDescriptor : origDescriptors) { final String name = origDescriptor.getName(); // Need to check isReadable() for WrapDynaBean // (see Jira issue# BEANUTILS-61) if (getPropertyUtils().isReadable(orig, name) && getPropertyUtils().isWriteable(dest, name)) { final Object value = ((DynaBean) orig).get(name); copyProperty(dest, name, value); } } } elseif (orig instanceof Map) { @SuppressWarnings('unchecked') final // Map properties are always of type <String, Object> Map<String, Object> propMap = (Map<String, Object>) orig; for (final Map.Entry<String, Object> entry : propMap.entrySet()) { final String name = entry.getKey(); if (getPropertyUtils().isWriteable(dest, name)) { copyProperty(dest, name, entry.getValue()); } } } else/* if (orig is a standard JavaBean) */ { final PropertyDescriptor[] origDescriptors = getPropertyUtils().getPropertyDescriptors(orig); for (PropertyDescriptor origDescriptor : origDescriptors) { final String name = origDescriptor.getName(); if ('class'.equals(name)) { continue; // No point in trying to set an object’s class } if (getPropertyUtils().isReadable(orig, name) && getPropertyUtils().isWriteable(dest, name)) { try { final Object value = getPropertyUtils().getSimpleProperty(orig, name); copyProperty(dest, name, value); } catch (final NoSuchMethodException e) { // Should not happen } } } } }

# Spring 的 BeanUtils

使用spring的BeanUtils進行對象拷貝:

publicclass TestSpringBeanUtils { public static void main(String[] args) throws InvocationTargetException, IllegalAccessException { //下面只是用于單獨測試 PersonSource personSource = new PersonSource(1, 'pjmike', '12345', 21); PersonDest personDest = new PersonDest(); BeanUtils.copyProperties(personSource,personDest); System.out.println('persondest: '+personDest); }}

Spring下的BeanUtils也是使用 copyProperties方法進行拷貝,只不過它的實現方式非常簡單,就是對兩個對象中相同名字的屬性進行簡單的get/set,僅檢查屬性的可訪問性。具體實現如下:

private static void copyProperties(Object source, Object target, @Nullable Class<?> editable, @Nullable String... ignoreProperties) throws BeansException { Assert.notNull(source, 'Source must not be null'); Assert.notNull(target, 'Target must not be null'); Class<?> actualEditable = target.getClass(); if (editable != null) { if (!editable.isInstance(target)) { throw new IllegalArgumentException('Target class [' + target.getClass().getName() + '] not assignable to Editable class [' + editable.getName() + ']'); } actualEditable = editable; } PropertyDescriptor[] targetPds = getPropertyDescriptors(actualEditable); List<String> ignoreList = (ignoreProperties != null ? Arrays.asList(ignoreProperties) : null); for (PropertyDescriptor targetPd : targetPds) { Method writeMethod = targetPd.getWriteMethod(); if (writeMethod != null && (ignoreList == null || !ignoreList.contains(targetPd.getName()))) { PropertyDescriptor sourcePd = getPropertyDescriptor(source.getClass(), targetPd.getName()); if (sourcePd != null) { Method readMethod = sourcePd.getReadMethod(); if (readMethod != null && ClassUtils.isAssignable(writeMethod.getParameterTypes()[0], readMethod.getReturnType())) { try { if (!Modifier.isPublic(readMethod.getDeclaringClass().getModifiers())) { readMethod.setAccessible(true); } Object value = readMethod.invoke(source); if (!Modifier.isPublic(writeMethod.getDeclaringClass().getModifiers())) { writeMethod.setAccessible(true); } writeMethod.invoke(target, value); } catch (Throwable ex) { throw new FatalBeanException( 'Could not copy property ’' + targetPd.getName() + '’ from source to target', ex); } } } } } }

可以看到,成員變量賦值是基于目標對象的成員列表,并且會跳過ignore的以及在源對象中不存在,所以這個方法是安全的,不會因為兩個對象之間的結構差異導致錯誤,但是必須保證同名的兩個成員變量類型相同

# 小結

以上簡要的分析兩種BeanUtils,因為Apache下的BeanUtils性能較差,不建議使用,可以使用 Spring的BeanUtils ,或者使用其他拷貝框架,比如:Dozer、ModelMapper等等

到此這篇關于淺析Java中Apache BeanUtils和Spring BeanUtils的用法的文章就介紹到這了,更多相關Apache BeanUtils和Spring BeanUtils內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
主站蜘蛛池模板: 欧美高清视频手机在在线 | 国产在线精品一区免费香蕉 | 一区二区三区欧美日韩国产 | 一级毛片视频免费观看 | 免费国产一区二区在免费观看 | 亚洲精品在线免费观看视频 | www.成人| 手机毛片免费看 | 成年性午夜免费视频网站不卡 | 欧美亚洲国产成人高清在线 | 视频在线色 | 国产韩国精品一区二区三区 | 男人的天堂免费网站 | 日本一级特黄毛片高清视频 | 香蕉久久久久久狠狠色 | 成人一区二区免费中文字幕 | 日本一区二区三区不卡在线视频 | 美女视频网站免费播放视 | 国产私拍福利精品视频推出 | 一区二区三区在线观看视频 | 国产在线观看91精品一区 | 亚洲免费成人在线 | 国产综合精品久久亚洲 | 成人免费在线观看视频 | 亚洲视频在线观看免费视频 | 国产亚洲综合精品一区二区三区 | 美女张开腿让我桶 | 亚洲欧美一区二区三区在线播放 | 国内精品一区二区 | 久久―日本道色综合久久 | avtom影院入口永久在线 | 欧美人成毛片在线播放 | 久久精品国产影库免费看 | 中国美女牲交一级毛片 | 日本二级毛片免费 | 国模偷拍在线观看免费视频 | 一级欧美毛片成人 | 国产91丝袜美腿在线观看 | 嫩模大尺度人体福利视频 | 国产精品理论 | 三级全黄视频 |