欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于Beanutils.copyProperties()的用法及重寫(xiě)提高效率

 更新時(shí)間:2021年09月01日 09:21:06   作者:菜鳥(niǎo)凱一枚  
這篇文章主要介紹了Beanutils.copyProperties( )的用法及重寫(xiě)提高效率的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

Beanutils.copyProperties()用法及重寫(xiě)提高效率

特別說(shuō)明本文介紹的是Spring(import org.springframework.beans.BeanUtils)中的BeanUtils.copyProperties(A,B)方法。是將A中的值賦給B。apache(org.apache.commons.beanutils.BeanUtils)中的BeanUtils.copyProperties(A,B)方法是將B中的值賦值給A。

一、簡(jiǎn)介

BeanUtils提供對(duì)Java反射和自省API的包裝。其主要目的是利用反射機(jī)制對(duì)JavaBean的屬性進(jìn)行處理。我們知道,一個(gè)JavaBean通常包含了大量的屬性,很多情況下,對(duì)JavaBean的處理導(dǎo)致大量get/set代碼堆積,增加了代碼長(zhǎng)度和閱讀代碼的難度。

二、用法

如果你有兩個(gè)具有很多相同屬性的JavaBean,一個(gè)很常見(jiàn)的情況就是Struts里的PO對(duì)象(持久對(duì)象)和對(duì)應(yīng)的ActionForm。例如:一個(gè)用戶注冊(cè)頁(yè)面,有一個(gè)User實(shí)體類和一個(gè)UserActionForm,我們一般會(huì)在Action里從ActionForm構(gòu)造一個(gè)PO對(duì)象,傳統(tǒng)的方式是使用類似下面的語(yǔ)句對(duì)屬性逐個(gè)賦值:

// 獲取 ActionForm 表單數(shù)據(jù)  
UserActionForm uForm = (UserActionForm) form; 
// 構(gòu)造一個(gè)User對(duì)象  
User user = new User();  
// 逐一賦值  
user.setUsername(uForm.getUsername);  
user.setPassword(uForm.getPassword);  
user.setAge(uForm.getAge);    
...........  
...........  
// 然后調(diào)用JDBC、或操作Hibernate 持久化對(duì)象User到數(shù)據(jù)庫(kù)  
HibernateDAO.save(user); 

通過(guò)這樣的方法如果表單數(shù)據(jù)N多、100、1000(夸張點(diǎn)。哈哈)、、、、那我們不是要寫(xiě)100、、、1000行set、get了。誰(shuí)都

不愿意這樣做。

而我們使用 BeanUtils.copyProperties() 方法以后,代碼量大大的減少,而且整體程序看著也簡(jiǎn)潔明朗,代碼如下:

// 獲取 ActionForm 表單數(shù)據(jù)  
UserActionForm uForm = (UserActionForm) form;      
// 構(gòu)造一個(gè)User對(duì)象  
User user = new User();    
BeanUtils.copyProperties(uForm,user);    
// 然后調(diào)用JDBC、或操作Hibernate 持久化對(duì)象User到數(shù)據(jù)庫(kù)  
HibernateDAO.save(user); 

注:如果User和UserActionForm 間存在名稱不相同的屬性,則BeanUtils不對(duì)這些屬性進(jìn)行處理,需要手動(dòng)處理。例如:

User類里面有個(gè)createDate 創(chuàng)建時(shí)間字段,而UserActionForm里面無(wú)此字段。BeanUtils.copyProperties()不會(huì)對(duì)此字段做任何處理。必須要自己手動(dòng)處理。

user.setModifyDate(new Date());  

三、重寫(xiě)

ReflectASM,高性能的反射:

什么是ReflectASM ReflectASM是一個(gè)很小的java類庫(kù),主要是通過(guò)asm生產(chǎn)類來(lái)實(shí)現(xiàn)java反射,執(zhí)行速度非???,看了網(wǎng)上很多和反射的對(duì)比,覺(jué)得ReflectASM比較神奇,很想知道其原理,下面介紹下如何使用及原理;

public static void main(String[] args) {    
        User user = new User();    
        //使用reflectasm生產(chǎn)User訪問(wèn)類    
        MethodAccess access = MethodAccess.get(User.class);    
        //invoke setName方法name值    
        access.invoke(user, "setName", "張三");    
        //invoke getName方法 獲得值    
        String name = (String)access.invoke(user, "getName", null);    
        System.out.println(name);    
    }    

原理

上面代碼的確實(shí)現(xiàn)反射的功能,代碼主要的核心是 MethodAccess.get(User.class);

看了下源碼,這段代碼主要是通過(guò)asm生產(chǎn)一個(gè)User的處理類 UserMethodAccess(這個(gè)類主要是實(shí)現(xiàn)了invoke方法)的ByteCode,然后獲得該對(duì)象,通過(guò)上面的invoke操作user類。

private static Map<Class, MethodAccess> methodMap = new HashMap<Class, MethodAccess>();    
    private static Map<String, Integer> methodIndexMap = new HashMap<String, Integer>(); 
    private static Map<Class, List<String>> fieldMap = new HashMap<Class, List<String>>();  
    public static void copyProperties(Object desc, Object orgi) {  
        MethodAccess descMethodAccess = methodMap.get(desc.getClass());  
        if (descMethodAccess == null) {  
            descMethodAccess = cache(desc);  
        }  
        MethodAccess orgiMethodAccess = methodMap.get(orgi.getClass());  
        if (orgiMethodAccess == null) {  
            orgiMethodAccess = cache(orgi);  
        }  
  
        List<String> fieldList = fieldMap.get(orgi.getClass());  
        for (String field : fieldList) {  
            String getKey = orgi.getClass().getName() + "." + "get" + field;  
            String setkey = desc.getClass().getName() + "." + "set" + field;  
            Integer setIndex = methodIndexMap.get(setkey);  
            if (setIndex != null) {  
                int getIndex = methodIndexMap.get(getKey);  
                // 參數(shù)一需要反射的對(duì)象  
                // 參數(shù)二class.getDeclaredMethods 對(duì)應(yīng)方法的index  
                // 參數(shù)對(duì)三象集合  
                descMethodAccess.invoke(desc, setIndex.intValue(),  
                        orgiMethodAccess.invoke(orgi, getIndex));  
            }  
        }  
    }  
  
    // 單例模式  
    private static MethodAccess cache(Object orgi) {  
        synchronized (orgi.getClass()) {  
            MethodAccess methodAccess = MethodAccess.get(orgi.getClass());  
            Field[] fields = orgi.getClass().getDeclaredFields();  
            List<String> fieldList = new ArrayList<String>(fields.length);  
            for (Field field : fields) {  
                if (Modifier.isPrivate(field.getModifiers())  
                        && !Modifier.isStatic(field.getModifiers())) { // 是否是私有的,是否是靜態(tài)的  
                    // 非公共私有變量  
                    String fieldName = StringUtils.capitalize(field.getName()); // 獲取屬性名稱  
                    int getIndex = methodAccess.getIndex("get" + fieldName); // 獲取get方法的下標(biāo)  
                    int setIndex = methodAccess.getIndex("set" + fieldName); // 獲取set方法的下標(biāo)  
                    methodIndexMap.put(orgi.getClass().getName() + "." + "get"  
                            + fieldName, getIndex); // 將類名get方法名,方法下標(biāo)注冊(cè)到map中  
                    methodIndexMap.put(orgi.getClass().getName() + "." + "set"  
                            + fieldName, setIndex); // 將類名set方法名,方法下標(biāo)注冊(cè)到map中  
                    fieldList.add(fieldName); // 將屬性名稱放入集合里  
                }  
            }  
            fieldMap.put(orgi.getClass(), fieldList); // 將類名,屬性名稱注冊(cè)到map中  
            methodMap.put(orgi.getClass(), methodAccess);  
            return methodAccess;  
        }  
    } 

執(zhí)行1000000條效率80幾毫秒,效率已經(jīng)沒(méi)問(wèn)題了。

BeanUtils.copyProperties 使用注意

首先結(jié)論說(shuō)在前頭, BeanUtils.copyProperties 是淺拷貝 。

為什么今天我還想把這個(gè)BeanUtils.copyProperties 的使用拿出來(lái)軍訓(xùn)。

因?yàn)槲乙庾R(shí)到了大家(部分)對(duì)深拷淺拷還是不清晰,不知道具體的影響。

示例演示

第一個(gè)類:

第二個(gè)類:

注意??! 第二個(gè)類里面有使用第一個(gè)類。

開(kāi)始進(jìn)行示例

    public static void main(String[] args) { 
        /**
         * 模擬數(shù)據(jù) A  complexObject
         */
        ComplexObject complexObjectA=new ComplexObject();
        complexObjectA.setNickName("張一");
        SimpleObject simpleObject=new SimpleObject();
        simpleObject.setName("李四");
        simpleObject.setAge(12);
        complexObjectA.setSimpleObject(simpleObject);  
        /**
         * 使用BeanUtils.copyProperties 拷貝 模擬數(shù)據(jù) A 生成模擬數(shù)據(jù) B
         */
        ComplexObject complexObjectB=new ComplexObject();
        BeanUtils.copyProperties(complexObjectA,complexObjectB);
 
        System.out.println("拷貝后,查看模擬數(shù)據(jù)A 和 模擬數(shù)據(jù)B :");
        System.out.println(complexObjectA.getSimpleObject().toString());
        System.out.println(complexObjectB.getSimpleObject().toString());
 
        System.out.println("比較模擬數(shù)據(jù)A 和 模擬數(shù)據(jù)B 里面的引用對(duì)象simple 是否引用地址一樣: ");
        System.out.println(complexObjectA.getSimpleObject()==complexObjectB.getSimpleObject()); 
 
        System.out.println("修改拷貝出來(lái)的模擬數(shù)據(jù)B里面的引用對(duì)象simple的屬性 age 為 888888");
        complexObjectB.getSimpleObject().setAge(888888);
 
        System.out.println("修改后,觀察原數(shù)據(jù)A 和拷貝出來(lái)的數(shù)據(jù) B 里面引用的 對(duì)象 simple的屬性 age:");
        System.out.println(complexObjectA.getSimpleObject().toString());
        System.out.println(complexObjectB.getSimpleObject().toString()); 
    }

最后強(qiáng)調(diào)

如果你是使用BeanUtils.copyProperties 進(jìn)行對(duì)象的拷貝復(fù)制, 一定要注意!

  • 第一點(diǎn) 、你所拷貝的對(duì)象內(nèi)包不包含 其他對(duì)象的引用。
  • 第二點(diǎn)、如果包含,那么接下來(lái)的方法里無(wú)論是操作原對(duì)象還是操作拷貝出來(lái)的對(duì)象是否涉及到 對(duì) 對(duì)象內(nèi) 的 那個(gè)其他對(duì)象的 值的修改 。
  • 第三點(diǎn)、如果涉及到, 修改了,會(huì)不會(huì)影響到其他方法 對(duì) 修改值的使用情況。

就如文中例子, 如果傳入過(guò)來(lái)的一個(gè)復(fù)雜對(duì)象數(shù)據(jù)A 里面引用了一個(gè) user對(duì)象年齡age是10;拷貝出一份數(shù)據(jù)B后, 操作 數(shù)據(jù)B的方法把 年齡age改成了88888;

那么后續(xù)其他方法用到數(shù)據(jù)A ,想用的是最初始的 age 為10 ,那么就用不到了,因?yàn)闇\拷貝的原因受影響,age都變成88888 了。

ok,就提到這。以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家~

相關(guān)文章

  • Java飛行記錄器JFR功能實(shí)現(xiàn)過(guò)程圖解

    Java飛行記錄器JFR功能實(shí)現(xiàn)過(guò)程圖解

    這篇文章主要介紹了Java飛行記錄器JFR功能實(shí)現(xiàn)過(guò)程圖解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-05-05
  • java根據(jù)List內(nèi)對(duì)象的屬性排序方法

    java根據(jù)List內(nèi)對(duì)象的屬性排序方法

    下面小編就為大家分享一篇java根據(jù)List內(nèi)對(duì)象的屬性排序方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2018-01-01
  • 徹底搞懂Java多線程(三)

    徹底搞懂Java多線程(三)

    這篇文章主要給大家介紹了關(guān)于Java面試題之多線程和高并發(fā)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-07-07
  • Java關(guān)鍵字之this用法詳解

    Java關(guān)鍵字之this用法詳解

    這篇文章將為大家詳細(xì)介紹一下Java關(guān)鍵字this的用法,文中有相關(guān)的代碼示例,希望對(duì)大家的學(xué)習(xí)或工作有一定的幫助,感興趣的同學(xué)可以參考下
    2023-05-05
  • Spring中的AutowireCandidateResolver的具體使用詳解

    Spring中的AutowireCandidateResolver的具體使用詳解

    這篇文章主要介紹了Spring中的AutowireCandidateResolver的具體使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-04-04
  • Java數(shù)據(jù)類型的規(guī)則

    Java數(shù)據(jù)類型的規(guī)則

    這篇文章主要介紹了Java數(shù)據(jù)類型的規(guī)則的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2016-12-12
  • 淺談Java多進(jìn)程程序的運(yùn)行模式

    淺談Java多進(jìn)程程序的運(yùn)行模式

    這篇文章主要介紹了淺談Java多進(jìn)程程序的運(yùn)行模式,包括對(duì)進(jìn)程阻塞問(wèn)題的討論等,需要的朋友可以參考下
    2015-11-11
  • Java超全面梳理內(nèi)部類的使用

    Java超全面梳理內(nèi)部類的使用

    說(shuō)起內(nèi)部類這個(gè)詞,想必很多人都不陌生,但是又會(huì)覺(jué)得不熟悉。原因是平時(shí)編寫(xiě)代碼時(shí)可能用到的場(chǎng)景不多,用得最多的是在有事件監(jiān)聽(tīng)的情況下,并且即使用到也很少去總結(jié)內(nèi)部類的用法。今天我們就來(lái)一探究竟
    2022-04-04
  • java中aop實(shí)現(xiàn)接口訪問(wèn)頻率限制

    java中aop實(shí)現(xiàn)接口訪問(wèn)頻率限制

    本文主要介紹了java中aop實(shí)現(xiàn)接口訪問(wèn)頻率限制,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • Java11新特性之HttpClient小試牛刀

    Java11新特性之HttpClient小試牛刀

    本文主要研究一下Java11的HttpClient的基本使用。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-09-09

最新評(píng)論