使用BeanUitls提高對象拷貝效率

對象拷貝

對象拷貝分為深拷貝和淺拷貝差凹。根據(jù)使用場景進行不同選擇期奔。在Java中,數(shù)據(jù)類型分為值類型(基本數(shù)據(jù)類型)和引用類型危尿,值類型包括int呐萌、double、byte谊娇、boolean肺孤、char等簡單數(shù)據(jù)類型,引用類型包括類济欢、接口赠堵、數(shù)組等復雜類型。

深度拷貝和淺度拷貝的主要區(qū)別在于是否支持引用類型的屬性拷貝法褥,本文將探討目前使用較多的幾種對象拷貝的方案茫叭,以及其是否支持深拷貝和性能對比。

關于深拷貝和淺拷貝的理解可以參考:http://www.reibang.com/p/e8c6155d9694

首先來創(chuàng)建兩個測試bean

注:一定要保證有set/get方法,成員變量必須要同名

@Data
public class User1 {
    String name;
    String password;
    String phone;
}
@Data
public class User2 {
    String name;
    String password;
    String phone;
}

1.Spring的BeanUtils(簡單易用)

org.springframework.beans.BeanUtils
BeanUtils.copyProperties(源對象半等,目標對象)

測試方法:
public static void main(String[] args){
        User1 user1=new User1();
        user1.setName("user1_name");
        user1.setPassword("user1_password");
        user1.setPhone("user1_phone");
        User2 user2=new User2();
        BeanUtils.copyProperties(user1,user2);
        System.out.println(user2.toString());
    }

執(zhí)行結果:User2(name=user1_name, password=user1_password, phone=user1_phone)

實現(xiàn)原理

BeanUtils部分源碼分析

public abstract class BeanUtils {
      public static void copyProperties(Object source, Object target) throws BeansException {
        copyProperties(source, target, (Class)null, (String[])null);
      }

      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;
        PropertyDescriptor[] var7 = targetPds;
        int var8 = targetPds.length;

        for(int var9 = 0; var9 < var8; ++var9) {
            PropertyDescriptor targetPd = var7[var9];
            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 var15) {
                            throw new FatalBeanException("Could not copy property '" + targetPd.getName() + "' from source to target", var15);
                        }
                    }
                }
            }
        }

    }
    ``````
}

實現(xiàn)的方式很簡單揍愁,就是對兩個對象中相同名字的屬性進行簡單get/set,僅檢查屬性的可訪問性酱鸭÷鹂澹可以看到, 成員變量賦值是基于目標對象的成員列表, 并且會跳過ignore的以及在源對象中不存在的, 所以這個方法是安全的, 不會因為兩個對象之間的結構差異導致錯誤

注:必須保證同名的兩個成員變量類型相同,同名屬性一個是包裝類型凹髓,一個是非包裝類型也是可以的

2.Apache的BeanUtils(拓展性強烁登,相對復雜)

Apache Common BeanUtil是一個常用的在對象之間復制數(shù)據(jù)的工具類,web開發(fā)框架struts就是依賴于它進行ActionForm的創(chuàng)建。
org.apache.commons.beanutils.BeanUtils
BeanUtils.copyProperties(目標對象饵沧,源對象)

需要引入依賴

        <dependency>
            <groupId>commons-beanutils</groupId>
            <artifactId>commons-beanutils</artifactId>
            <version>1.9.3</version>
        </dependency>
測試方法:
public static void main(String[] args) throws InvocationTargetException, IllegalAccessException {
        User1 user1=new User1();
        user1.setName("user1_name");
        user1.setPassword("user1_password");
        user1.setPhone("user1_phone");
        User2 user2=new User2();
        BeanUtils.copyProperties(user2,user1);
        System.out.println(user2.toString());
    }

執(zhí)行結果:User2(name=user1_name, password=user1_password, phone=user1_phone)

實現(xiàn)原理

BeanUtils部分源碼分析

public class BeanUtils {
    public static void copyProperties(Object dest, Object orig) throws IllegalAccessException, InvocationTargetException {
        BeanUtilsBean.getInstance().copyProperties(dest, orig);
    }

    public void copyProperties(Object dest, Object orig) throws IllegalAccessException, InvocationTargetException {
        if (dest == null) {
            throw new IllegalArgumentException("No destination bean specified");
        } else if (orig == null) {
            throw new IllegalArgumentException("No origin bean specified");
        } else {
            if (this.log.isDebugEnabled()) {
                this.log.debug("BeanUtils.copyProperties(" + dest + ", " + orig + ")");
            }

            int i;
            String name;
            Object value;
            if (orig instanceof DynaBean) {
                DynaProperty[] origDescriptors = ((DynaBean)orig).getDynaClass().getDynaProperties();

                for(i = 0; i < origDescriptors.length; ++i) {
                    name = origDescriptors[i].getName();
                    if (this.getPropertyUtils().isReadable(orig, name) && this.getPropertyUtils().isWriteable(dest, name)) {
                        value = ((DynaBean)orig).get(name);
                        this.copyProperty(dest, name, value);
                    }
                }
            } else if (orig instanceof Map) {
                Iterator entries = ((Map)orig).entrySet().iterator();

                while(entries.hasNext()) {
                    Entry entry = (Entry)entries.next();
                    name = (String)entry.getKey();
                    if (this.getPropertyUtils().isWriteable(dest, name)) {
                        this.copyProperty(dest, name, entry.getValue());
                    }
                }
            } else {
                PropertyDescriptor[] origDescriptors = this.getPropertyUtils().getPropertyDescriptors(orig);

                for(i = 0; i < origDescriptors.length; ++i) {
                    name = origDescriptors[i].getName();
                    if (!"class".equals(name) && this.getPropertyUtils().isReadable(orig, name) && this.getPropertyUtils().isWriteable(dest, name)) {
                        try {
                            value = this.getPropertyUtils().getSimpleProperty(orig, name);
                            this.copyProperty(dest, name, value);
                        } catch (NoSuchMethodException var7) {
                        }
                    }
                }
            }

        }
    }
    ``````
}

commons-beanutils則施加了很多的檢驗锨络,包括類型的轉(zhuǎn)換,甚至于還會檢驗對象所屬的類的可訪問性狼牺。BeanUtils能夠順利的完成對象屬性值的復制羡儿,依賴于其對類型的識別。

除了支持基本類型以及基本類型的數(shù)組之外是钥,還支持java.sql.Date,java.sql.Time, java.sql.TimeStamp, java.io.File, javaio.URL這些類的對象掠归,其余一概不支持。不過可以自定義Converter悄泥。然后注冊進去虏冻。在org.apache.commons.beanutils.converters包中有一系列converter類,用于不同類型之間對象的轉(zhuǎn)化弹囚。主要通過注入新的類型轉(zhuǎn)換器厨相,因為默認情況下,BeanUtils對復雜對象的復制是引用

注:commons-beanutils中的裝換是不支持java.util.Date的鸥鹉。

BeanUtils的官方API文檔:https://commons.apache.org/proper/commons-beanutils/apidocs/org/apache/commons/beanutils/BeanUtils.html
converter的官方API文檔:https://commons.apache.org/proper/commons-beanutils/apidocs/org/apache/commons/beanutils/Converter.html
BeanUtils自定義的轉(zhuǎn)換器可以參考
https://blog.csdn.net/marksinoberg/article/details/51830076
https://blog.csdn.net/qq_14945847/article/details/77450222

總結:關于bean復制蛮穿,如果屬性較少,建議直接寫個方法完成get/set即可毁渗。如果屬性較多践磅,可以自己采用反射實現(xiàn)一個滿足自己需要的工具類,或者使用BeanUtils類祝蝠。BeanUtils是利用反射機制對JavaBean的屬性進行處理音诈。一個JavaBean通常包含了大量的屬性幻碱,很多情況下绎狭,對JavaBean的處理導致大量get/set代碼堆積,增加了代碼長度和閱讀代碼的難度褥傍。由于這些BeanUtils類都是采用反射機制實現(xiàn)的儡嘶,對程序的效率會有影響。
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末恍风,一起剝皮案震驚了整個濱河市蹦狂,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌朋贬,老刑警劉巖凯楔,帶你破解...
    沈念sama閱讀 211,743評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異锦募,居然都是意外死亡摆屯,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,296評論 3 385
  • 文/潘曉璐 我一進店門糠亩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來虐骑,“玉大人准验,你說我怎么就攤上這事⊥⒚唬” “怎么了糊饱?”我有些...
    開封第一講書人閱讀 157,285評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長颠黎。 經(jīng)常有香客問我另锋,道長,這世上最難降的妖魔是什么狭归? 我笑而不...
    開封第一講書人閱讀 56,485評論 1 283
  • 正文 為了忘掉前任砰蠢,我火速辦了婚禮,結果婚禮上唉铜,老公的妹妹穿的比我還像新娘台舱。我一直安慰自己,他們只是感情好潭流,可當我...
    茶點故事閱讀 65,581評論 6 386
  • 文/花漫 我一把揭開白布竞惋。 她就那樣靜靜地躺著,像睡著了一般灰嫉。 火紅的嫁衣襯著肌膚如雪拆宛。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,821評論 1 290
  • 那天讼撒,我揣著相機與錄音浑厚,去河邊找鬼。 笑死根盒,一個胖子當著我的面吹牛钳幅,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播炎滞,決...
    沈念sama閱讀 38,960評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼敢艰,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了册赛?” 一聲冷哼從身側響起钠导,我...
    開封第一講書人閱讀 37,719評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎森瘪,沒想到半個月后牡属,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,186評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡扼睬,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,516評論 2 327
  • 正文 我和宋清朗相戀三年逮栅,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,650評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡证芭,死狀恐怖瞳浦,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情废士,我是刑警寧澤叫潦,帶...
    沈念sama閱讀 34,329評論 4 330
  • 正文 年R本政府宣布,位于F島的核電站官硝,受9級特大地震影響矗蕊,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜氢架,卻給世界環(huán)境...
    茶點故事閱讀 39,936評論 3 313
  • 文/蒙蒙 一傻咖、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧岖研,春花似錦卿操、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,757評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至拓售,卻和暖如春窥摄,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背础淤。 一陣腳步聲響...
    開封第一講書人閱讀 31,991評論 1 266
  • 我被黑心中介騙來泰國打工崭放, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人鸽凶。 一個月前我還...
    沈念sama閱讀 46,370評論 2 360
  • 正文 我出身青樓币砂,卻偏偏與公主長得像,于是被迫代替她去往敵國和親吱瘩。 傳聞我的和親對象是個殘疾皇子道伟,可洞房花燭夜當晚...
    茶點故事閱讀 43,527評論 2 349

推薦閱讀更多精彩內(nèi)容