chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

為什么有時(shí)候會(huì)寫(xiě)出爛代碼

深圳東裕光大 ? 來(lái)源:Hollis ? 作者:Hollis ? 2021-08-27 10:23 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

本文的內(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í)了~就容易被忽略了,于是我在團(tuán)隊(duì)群里面強(qiáng)調(diào)了一下這個(gè)問(wèn)題:

所以,本文主要是關(guān)于BeanUtils工具的屬性拷貝以及深拷貝、淺拷貝等問(wèn)題的。好了開(kāi)始正文,介紹下問(wèn)題代碼是什么,為什么有問(wèn)題,又符合修改?

在日常開(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中的大部分屬性都是一樣的,所以使用屬性拷貝類工具可以幫助我們節(jié)省大量的set和get操作。

市面上有很多類似的工具類,比較常用的有

1、Spring BeanUtils

2、Cglib BeanCopier

3、Apache BeanUtils

4、Apache PropertyUtils

5、Dozer

6、MapStucts

這里面我比較建議大家使用的是MapStructs。

最近我們有個(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ù)類型進(jìn)行值傳遞,對(duì)引用數(shù)據(jù)類型進(jìn)行引用傳遞般的拷貝,此為淺拷貝。

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

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

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

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ì)象也受到了修改的影響。

這就是所謂的淺拷貝!

74732e18-f46d-11eb-9bcf-12bb97331649.png

如何進(jìn)行深拷貝

發(fā)現(xiàn)問(wèn)題之后,我們就要想辦法解決,那么如何實(shí)現(xiàn)深拷貝呢?

1、實(shí)現(xiàn)Cloneable接口,重寫(xiě)clone()

在Object類中定義了一個(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類,使他們實(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)我們使用各類BeanUtils的時(shí)候,一定要注意是淺拷貝還是深拷貝,淺拷貝的結(jié)果就是兩個(gè)對(duì)象中的引用對(duì)象都是同一個(gè)地址,只要發(fā)生改變,都會(huì)有影響。

想要實(shí)現(xiàn)深拷貝,有很多種辦法,其中比較常用的就是實(shí)現(xiàn)Cloneable接口重寫(xiě)clone方法,還有使用序列化+反序列化創(chuàng)建新對(duì)象。

好了,以上就是今天的全部?jī)?nèi)容了。

責(zé)任編輯:haq

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4900

    瀏覽量

    70754
  • 工具
    +關(guān)注

    關(guān)注

    4

    文章

    315

    瀏覽量

    28293

原文標(biāo)題:這樣的爛代碼,我實(shí)習(xí)的時(shí)候都寫(xiě)不出來(lái)!

文章出處:【微信號(hào):sztonyu,微信公眾號(hào):深圳東裕光大】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    用ADS1256測(cè)微安級(jí)電流,采樣電阻用的3.3歐姆,有時(shí)候會(huì)出現(xiàn)得到的AD值不準(zhǔn),為什么?

    用ADS1256測(cè)微安級(jí)電流,采樣電阻用的3.3歐姆,有時(shí)候會(huì)出現(xiàn)得到的AD值不準(zhǔn),應(yīng)該是干擾的原因,我自己調(diào)試的時(shí)候沒(méi)出現(xiàn)這個(gè)問(wèn)題,在車間里出現(xiàn)了這個(gè)問(wèn)題,用手摸摸采樣電路,有時(shí)候會(huì)恢復(fù)正常。求解??? 謝謝!
    發(fā)表于 02-12 06:28

    把兩個(gè)DAC121S101連接成菊花鏈結(jié)構(gòu),為什么第二片的輸出有時(shí)候不是想要的電壓?

    我把兩個(gè)DAC121S101連接成菊花鏈結(jié)構(gòu),為什么第二片的輸出有時(shí)候不是我想要的電壓?
    發(fā)表于 01-23 07:32

    TCA8418用來(lái)做鍵盤(pán)擴(kuò)展,有時(shí)候設(shè)備開(kāi)關(guān)機(jī)或進(jìn)入睡眠狀態(tài),啟動(dòng)或喚醒會(huì)出現(xiàn)鍵盤(pán)失靈現(xiàn)象,怎么解決?

    在TCA8418旁邊有13.56MHZ的讀卡模塊,TCA8418用來(lái)做鍵盤(pán)擴(kuò)展,有時(shí)候設(shè)備開(kāi)關(guān)機(jī),或進(jìn)入睡眠狀態(tài),啟動(dòng)或喚醒會(huì)出現(xiàn)鍵盤(pán)失靈現(xiàn)象。8418是什么原因受干擾,有什么辦法避免么,有什么好的解決方法。
    發(fā)表于 01-03 08:04

    ADS1191采集內(nèi)部測(cè)試信TEST1Hz方波,有時(shí)候hen正常有時(shí)候不正常,是哪里出了問(wèn)題?

    我采集內(nèi)部測(cè)試信TEST1Hz方波,有時(shí)候hen正常 有時(shí)候不正常,不知道是哪里出現(xiàn)問(wèn)題 很明顯第一幅圖是比較正常的 第二幅會(huì)出現(xiàn)問(wèn)題。有沒(méi)有大神知道這個(gè)問(wèn)題是什么原因 這是我
    發(fā)表于 12-30 06:30

    在上電后,有時(shí)候會(huì)出現(xiàn)DAC1220E輸出不清零的情況,怎么解決?

    ,DAC1220E電路圖均按照規(guī)格書(shū)基本電路設(shè)計(jì),周圍電容均為NPO材質(zhì)的,在使用過(guò)程中出現(xiàn)以下問(wèn)題: 一、在上電后,有時(shí)候會(huì)出現(xiàn)DAC1220E輸出不清零,為2.5V輸出; 二、一旦出現(xiàn)上電不清
    發(fā)表于 12-18 07:24

    ADS1282讀寫(xiě)寄存器讀不出正確的結(jié)果,有時(shí)候是全0,為什么?

    的什么值,再回讀寄存器內(nèi)容的時(shí)候就讀不出正確的結(jié)果,有時(shí)候是全0,有時(shí)候是前幾個(gè)寄存器值對(duì)而后邊的不對(duì),還有完全不對(duì)的情況。讀采集數(shù)據(jù)也是無(wú)規(guī)律的亂變的數(shù)值或者全0或者滿量程值等錯(cuò)誤數(shù)據(jù),與實(shí)際輸入
    發(fā)表于 12-13 06:15

    使用DAC5675A,有時(shí)候DA輸出有毛刺,和兩個(gè)時(shí)鐘引腳的差值400mv-800mv不滿足有關(guān)系嗎?

    。有時(shí)候DA輸出有毛刺,和兩個(gè)時(shí)鐘引腳的差值400mv-800mv不滿足有關(guān)系嗎??jī)蓚€(gè)時(shí)鐘引腳的差值400~800mv是對(duì)輸入差分時(shí)鐘的限制嗎?還是輸入CMOS電平也有要求
    發(fā)表于 12-04 08:29

    DAC8563xa中的10腳原本是輸出2.5v電壓,現(xiàn)在有時(shí)候有輸出2.5v,有時(shí)候沒(méi)有,為什么?

    DAC8563xa中的10腳原本是輸出2.5v電壓,現(xiàn)在有時(shí)候有輸出2.5v,有時(shí)候沒(méi)有,是不是基本上能確定是芯片壞了
    發(fā)表于 11-28 07:22

    如何寫(xiě)出穩(wěn)定的單片機(jī)代碼

    這篇文章分享怎么寫(xiě)出穩(wěn)定的單片機(jī)代碼。? ?? 我對(duì)優(yōu)秀代碼的理解,大體分為兩個(gè)部分:高效和穩(wěn)定。 ? 兩者都能做到很好的,如果靠自己摸索,沒(méi)有刻意去練習(xí),可能需要花10年,甚至更久
    的頭像 發(fā)表于 11-15 16:40 ?825次閱讀
    如何<b class='flag-5'>寫(xiě)出</b>穩(wěn)定的單片機(jī)<b class='flag-5'>代碼</b>

    LMX2595RHAR有時(shí)候鎖不住頻怎么回事?

    LMX2595RHAR 有時(shí)候鎖不住頻怎么回事? 6G的信號(hào)有時(shí)能鎖住,有時(shí)又鎖不住,但是我們用的另外一個(gè)頻點(diǎn),4.5G,又沒(méi)這個(gè)現(xiàn)象 鎖不住的時(shí)候就是上圖那樣的,它的VT信號(hào)也是一
    發(fā)表于 11-11 06:38

    TAS5711有時(shí)候出現(xiàn)PLL autolock err,為什么?

    TAS5711有時(shí)候出現(xiàn)PLL autolock err
    發(fā)表于 10-30 06:02

    TAS5719調(diào)試DEMO板,有時(shí)候輸出幅度會(huì)不一樣,為什么?

    我想了解TI的TAS5719的壓限功能。并且我發(fā)現(xiàn)調(diào)試DEMO板,有時(shí)候輸出幅度會(huì)不一樣!比如,第一次開(kāi)機(jī)我可以輸出電平可以壓在4V左右,但我僅僅調(diào)試音量,有可以輸出6V-7V.輸出信號(hào)波形失真了!
    發(fā)表于 10-15 08:19

    TLC27M9有時(shí)候不起振是哪里出了問(wèn)題?

    有個(gè)儀器使用了電子渦流側(cè)厚度的原理。模擬電路用了TLC27M9C或者TLC27M7C兩種方案。 TLC27M9C會(huì)出現(xiàn)測(cè)量過(guò)程中 方波信號(hào)突然沒(méi)有的現(xiàn)象,更換好多批次芯片有時(shí)候這個(gè)現(xiàn)象就改善了
    發(fā)表于 08-30 06:16

    運(yùn)放單電源供電,示波器直流耦合檔測(cè)試,為什么有時(shí)候示波器測(cè)的輸出信號(hào)出現(xiàn)負(fù)值?

    運(yùn)放單電源供電,示波器直流耦合檔測(cè)試,為什么有時(shí)候示波器測(cè)的輸出信號(hào)出現(xiàn)負(fù)值?請(qǐng)解答,謝謝!
    發(fā)表于 08-29 08:25

    LTM8053,-12V輸出,連續(xù)開(kāi)關(guān)機(jī),有時(shí)候啟動(dòng)不了是什么原因?

    LTM 8053,-12V輸出,連續(xù)開(kāi)關(guān)機(jī),有時(shí)候啟動(dòng)不了,是什么原因?
    發(fā)表于 07-24 07:30