1. <strong id="7actg"></strong>
    2. <table id="7actg"></table>

    3. <address id="7actg"></address>
      <address id="7actg"></address>
      1. <object id="7actg"><tt id="7actg"></tt></object>

        這樣的爛代碼,我實(shí)習(xí)的時(shí)候都寫(xiě)不出來(lái)!

        共 5626字,需瀏覽 12分鐘

         ·

        2021-07-31 18:41

        作者 l Hollis
        來(lái)源 l Hollis(ID:hollischuang)

        本文的內(nèi)容是最近我剛剛遇到的一個(gè)問(wèn)題,問(wèn)題代碼是我自己寫(xiě)的,也是我自己寫(xiě)單元測(cè)試的時(shí)候發(fā)現(xiàn)的,也是我自己修復(fù)的,修復(fù)完之后,我反思了一下:這樣的問(wèn)題代碼,我實(shí)習(xí)的時(shí)候都寫(xiě)不出來(lái)。
        可是為什么我就寫(xiě)出來(lái)了呢?其實(shí)還是因?yàn)橛行┲R(shí)沒(méi)那么扎實(shí)了~就容易被忽略了
        在日常開(kāi)發(fā)中,我們經(jīng)常需要給對(duì)象進(jìn)行賦值,通常會(huì)調(diào)用其set/get方法,有些時(shí)候,如果我們要轉(zhuǎn)換的兩個(gè)對(duì)象之間屬性大致相同,會(huì)考慮使用屬性拷貝工具進(jìn)行。
        如我們經(jīng)常在代碼中會(huì)對(duì)一個(gè)數(shù)據(jù)結(jié)構(gòu)封裝成DO、SDO、DTO、VO等,而這些Bean中的大部分屬性都是一樣的,所以使用屬性拷貝類(lèi)工具可以幫助我們節(jié)省大量的set和get操作。
        市面上有很多類(lèi)似的工具類(lèi),比較常用的有
        1、Spring BeanUtils
        2、Cglib BeanCopier
        3、Apache BeanUtils
        4、Apache PropertyUtils
        5、Dozer
        6、MapStucts
        這里面我比較建議大家使用的是MapStructs,我在《丟棄掉那些BeanUtils工具類(lèi)吧,MapStruct真香?。?!》中介紹過(guò)原因。這里就不再贅述了。
        最近我們有個(gè)新項(xiàng)目,要?jiǎng)?chuàng)建一個(gè)新的應(yīng)用,因?yàn)槲易约悍治鲞^(guò)這些工具的效率,也去看過(guò)他們的實(shí)現(xiàn)原理,比較下來(lái)之后,我覺(jué)得MapStruct是最適合我們的,于是就在代碼中引入了這個(gè)框架。
        另外,因?yàn)镾pring的BeanUtils用起來(lái)也比較方便,所以,代碼中對(duì)于需要beanCopy的地方主要在使用這兩個(gè)框架。
        我們一般是這樣的,如果是DO和DTO/Entity之間的轉(zhuǎn)換,我們統(tǒng)一使用MapStruct,因?yàn)樗梢灾付▎为?dú)的Mapper,可以自定義一些策略。
        如果是同對(duì)象之間的拷貝(如用一個(gè)DO創(chuàng)建一個(gè)新的DO),或者完全不相關(guān)的兩個(gè)對(duì)象轉(zhuǎn)換,則使用Spring的BeanUtils。
        剛開(kāi)始都沒(méi)什么問(wèn)題,但是后面我在寫(xiě)單測(cè)的時(shí)候,發(fā)現(xiàn)了一個(gè)問(wèn)題。



        問(wèn)題
        先來(lái)看看我們是在什么地方用的Spring的BeanUtils
        我們的業(yè)務(wù)邏輯中,需要對(duì)訂單信息進(jìn)行修改,在更改時(shí),不僅要更新訂單的上面的屬性信息,還需要?jiǎng)?chuàng)建一條變更流水。
        而變更流水中同時(shí)記錄了變更前和變更后的數(shù)據(jù),所以就有了以下代碼:

        //從數(shù)據(jù)庫(kù)中查詢出當(dāng)前訂單,并加鎖

        OrderDetail orderDetail = orderDetailDao.queryForLock();


        //copy一個(gè)新的訂單模型

        OrderDetail newOrderDetail = new OrderDetail();

        BeanUtils.copyProperties(orderDetail, newOrderDetail);


        //對(duì)新的訂單模型進(jìn)行修改邏輯操作

        newOrderDetail.update();


        //使用修改前的訂單模型和修改后的訂單模型組裝出訂單變更流水

        OrderDetailStream orderDetailStream = new OrderDetailStream();

        orderDetailStream.create(orderDetail, newOrderDetail);

        大致邏輯是這樣的,因?yàn)閯?chuàng)建訂單變更流水的時(shí)候,需要一個(gè)改變前的訂單和改變后的訂單。所以我們想到了要new一個(gè)新的訂單模型,然后操作新的訂單模型,避免對(duì)舊的有影響。
        但是,就是這個(gè)BeanUtils.copyProperties的過(guò)程其實(shí)是有問(wèn)題的。
        因?yàn)锽eanUtils在進(jìn)行屬性copy的時(shí)候,本質(zhì)上是淺拷貝,而不是深拷貝。



        淺拷貝?深拷貝?
        什么是淺拷貝和深拷貝?來(lái)看下概念。
        1、淺拷貝:對(duì)基本數(shù)據(jù)類(lèi)型進(jìn)行值傳遞,對(duì)引用數(shù)據(jù)類(lèi)型進(jìn)行引用傳遞般的拷貝,此為淺拷貝。

        2、深拷貝:對(duì)基本數(shù)據(jù)類(lèi)型進(jìn)行值傳遞,對(duì)引用數(shù)據(jù)類(lèi)型,創(chuàng)建一個(gè)新的對(duì)象,并復(fù)制其內(nèi)容,此為深拷貝。

        我們舉個(gè)實(shí)際例子,來(lái)看下為啥我說(shuō)BeanUtils.copyProperties的過(guò)程是淺拷貝。

        先來(lái)定義兩個(gè)類(lèi):

        public class Address {

            private String province;

            private String city;

            private String area;

            //省略構(gòu)造函數(shù)和setter/getter

        }


        class User {

            private String name;

            private String password;

            private Address address;

            //省略構(gòu)造函數(shù)和setter/getter

        }

        然后寫(xiě)一段測(cè)試代碼:

        User user = new User("Hollis""hollischuang");

        user.setAddress(new Address("zhejiang""hangzhou""binjiang"));


        User newUser = new User();

        BeanUtils.copyProperties(user, newUser);

        System.out.println(user.getAddress() == newUser.getAddress());

        以上代碼輸出結(jié)果為:true
        即,我們BeanUtils.copyProperties拷貝出來(lái)的newUser中的address對(duì)象和原來(lái)的user中的address對(duì)象是同一個(gè)對(duì)象。
        可以嘗試著修改下newUser中的address對(duì)象:

            newUser.getAddress().setCity("shanghai");

            System.out.println(JSON.toJSONString(user));

            System.out.println(JSON.toJSONString(newUser));

        輸出結(jié)果:

        {"address":{"area":"binjiang","city":"shanghai","province":"zhejiang"},"name":"Hollis","password":"hollischuang"}

        {"address":{"area":"binjiang","city":"shanghai","province":"zhejiang"},"name":"Hollis","password":"hollischuang"}

        可以發(fā)現(xiàn),原來(lái)的對(duì)象也受到了修改的影響。
        這就是所謂的淺拷貝!



        如何進(jìn)行深拷貝
        發(fā)現(xiàn)問(wèn)題之后,我們就要想辦法解決,那么如何實(shí)現(xiàn)深拷貝呢?
        1、實(shí)現(xiàn)Cloneable接口,重寫(xiě)clone()
        在Object類(lèi)中定義了一個(gè)clone方法,這個(gè)方法其實(shí)在不重寫(xiě)的情況下,其實(shí)也是淺拷貝的。
        如果想要實(shí)現(xiàn)深拷貝,就需要重寫(xiě)clone方法,而想要重寫(xiě)clone方法,就必須實(shí)現(xiàn)Cloneable,否則會(huì)報(bào)CloneNotSupportedException異常。
        將上述代碼修改下,重寫(xiě)clone方法:

        public class Address implements Cloneable{

            private String province;

            private String city;

            private String area;

            //省略構(gòu)造函數(shù)和setter/getter



            @Override

            public Object clone() throws CloneNotSupportedException {

                return super.clone();

            }

        }


        class User implements Cloneable{

            private String name;

            private String password;

            private Address address;

            //省略構(gòu)造函數(shù)和setter/getter


            @Override

            protected Object clone() throws CloneNotSupportedException {

                User user = (User)super.clone();

                user.setAddress((Address)address.clone());

                return user;

            }

        }

        之后,在執(zhí)行一下上面的測(cè)試代碼,就可以發(fā)現(xiàn),這時(shí)候newUser中的address對(duì)象就是一個(gè)新的對(duì)象了。
        這種方式就能實(shí)現(xiàn)深拷貝,但是問(wèn)題是如果我們?cè)赨ser中有很多個(gè)對(duì)象,那么clone方法就寫(xiě)的很長(zhǎng),而且如果后面有修改,在User中新增屬性,這個(gè)地方也要改。
        那么,有沒(méi)有什么辦法可以不需要修改,一勞永逸呢?
        2、序列化實(shí)現(xiàn)深拷貝
        我們可以借助序列化來(lái)實(shí)現(xiàn)深拷貝。先把對(duì)象序列化成流,再?gòu)牧髦蟹葱蛄谢蓪?duì)象,這樣就一定是新的對(duì)象了。
        序列化的方式有很多,比如我們可以使用各種JSON工具,把對(duì)象序列化成JSON字符串,然后再?gòu)淖址蟹葱蛄谢蓪?duì)象。
        如使用fastjson實(shí)現(xiàn):

        User newUser = JSON.parseObject(JSON.toJSONString(user), User.class);

        也可實(shí)現(xiàn)深拷貝。
        除此之外,還可以使用Apache Commons Lang中提供的SerializationUtils工具實(shí)現(xiàn)。
        我們需要修改下上面的User和Address類(lèi),使他們實(shí)現(xiàn)Serializable接口,否則是無(wú)法進(jìn)行序列化的。

        class User implements Serializable

        class Address implements Serializable

        然后在需要拷貝的時(shí)候:

        User newUser = (User) SerializationUtils.clone(user);

        同樣,也可以實(shí)現(xiàn)深拷貝啦~!


        總結(jié)
        當(dāng)我們使用各類(lèi)BeanUtils的時(shí)候,一定要注意是淺拷貝還是深拷貝,淺拷貝的結(jié)果就是兩個(gè)對(duì)象中的引用對(duì)象都是同一個(gè)地址,只要發(fā)生改變,都會(huì)有影響。
        想要實(shí)現(xiàn)深拷貝,有很多種辦法,其中比較常用的就是實(shí)現(xiàn)Cloneable接口重寫(xiě)clone方法,還有使用序列化+反序列化創(chuàng)建新對(duì)象。
        好了,以上就是今天的全部?jī)?nèi)容了。

        推薦閱讀:


        喜歡我可以給我設(shè)為星標(biāo)哦

        好文章,我“在看”
        瀏覽 67
        點(diǎn)贊
        評(píng)論
        收藏
        分享

        手機(jī)掃一掃分享

        分享
        舉報(bào)
        評(píng)論
        圖片
        表情
        推薦
        點(diǎn)贊
        評(píng)論
        收藏
        分享

        手機(jī)掃一掃分享

        分享
        舉報(bào)
        1. <strong id="7actg"></strong>
        2. <table id="7actg"></table>

        3. <address id="7actg"></address>
          <address id="7actg"></address>
          1. <object id="7actg"><tt id="7actg"></tt></object>
            影音先锋无人区 | 欧美老妇婆性爱第一页 | 亚洲性爱电影院 | 少妇做爰毛片A片成人影院 | 无码AV在线免费观看 | 美女色黄 | 一本大道AV伊人久久综合蜜芽 | 日日澡韩国电影 | 女人被弄到高潮的视频 | 伦片丰满丰满午夜电影 |