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

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

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

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

為什么有時候會寫出爛代碼

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

掃碼添加小助手

加入工程師交流群

本文的內(nèi)容是最近我剛剛遇到的一個問題,問題代碼是我自己寫的,也是我自己寫單元測試的時候發(fā)現(xiàn)的,也是我自己修復(fù)的,修復(fù)完之后,我反思了一下:這樣的問題代碼,我實習(xí)的時候都寫不出來。

可是為什么我就寫出來了呢?其實還是因為有些知識沒那么扎實了~就容易被忽略了,于是我在團隊群里面強調(diào)了一下這個問題:

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

在日常開發(fā)中,我們經(jīng)常需要給對象進行賦值,通常會調(diào)用其set/get方法,有些時候,如果我們要轉(zhuǎn)換的兩個對象之間屬性大致相同,會考慮使用屬性拷貝工具進行。

如我們經(jīng)常在代碼中會對一個數(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。

最近我們有個新項目,要創(chuàng)建一個新的應(yīng)用,因為我自己分析過這些工具的效率,也去看過他們的實現(xiàn)原理,比較下來之后,我覺得MapStruct是最適合我們的,于是就在代碼中引入了這個框架。

另外,因為Spring的BeanUtils用起來也比較方便,所以,代碼中對于需要beanCopy的地方主要在使用這兩個框架。

我們一般是這樣的,如果是DO和DTO/Entity之間的轉(zhuǎn)換,我們統(tǒng)一使用MapStruct,因為他可以指定單獨的Mapper,可以自定義一些策略。

如果是同對象之間的拷貝(如用一個DO創(chuàng)建一個新的DO),或者完全不相關(guān)的兩個對象轉(zhuǎn)換,則使用Spring的BeanUtils。

剛開始都沒什么問題,但是后面我在寫單測的時候,發(fā)現(xiàn)了一個問題。

問題

先來看看我們是在什么地方用的Spring的BeanUtils

我們的業(yè)務(wù)邏輯中,需要對訂單信息進行修改,在更改時,不僅要更新訂單的上面的屬性信息,還需要創(chuàng)建一條變更流水。

而變更流水中同時記錄了變更前和變更后的數(shù)據(jù),所以就有了以下代碼:

//從數(shù)據(jù)庫中查詢出當前訂單,并加鎖 OrderDetail orderDetail = orderDetailDao.queryForLock(); //copy一個新的訂單模型 OrderDetail newOrderDetail = new OrderDetail();

BeanUtils.copyProperties(orderDetail, newOrderDetail);//對新的訂單模型進行修改邏輯操作

newOrderDetail.update(); //使用修改前的訂單模型和修改后的訂單模型組裝出訂單變更流水

OrderDetailStream orderDetailStream = new OrderDetailStream(); orderDetailStream.create(orderDetail, newOrderDetail);

大致邏輯是這樣的,因為創(chuàng)建訂單變更流水的時候,需要一個改變前的訂單和改變后的訂單。所以我們想到了要new一個新的訂單模型,然后操作新的訂單模型,避免對舊的有影響。

但是,就是這個BeanUtils.copyProperties的過程其實是有問題的。

因為BeanUtils在進行屬性copy的時候,本質(zhì)上是淺拷貝,而不是深拷貝。

淺拷貝?深拷貝?

什么是淺拷貝和深拷貝?來看下概念。

1、淺拷貝:對基本數(shù)據(jù)類型進行值傳遞,對引用數(shù)據(jù)類型進行引用傳遞般的拷貝,此為淺拷貝。

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

我們舉個實際例子,來看下為啥我說BeanUtils.copyProperties的過程是淺拷貝。

先來定義兩個類:

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 }

然后寫一段測試代碼:

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拷貝出來的newUser中的address對象和原來的user中的address對象是同一個對象。

可以嘗試著修改下newUser中的address對象:

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

這就是所謂的淺拷貝!

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

如何進行深拷貝

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

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

在Object類中定義了一個clone方法,這個方法其實在不重寫的情況下,其實也是淺拷貝的。

如果想要實現(xiàn)深拷貝,就需要重寫clone方法,而想要重寫clone方法,就必須實現(xiàn)Cloneable,否則會報CloneNotSupportedException異常。

將上述代碼修改下,重寫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í)行一下上面的測試代碼,就可以發(fā)現(xiàn),這時候newUser中的address對象就是一個新的對象了。

這種方式就能實現(xiàn)深拷貝,但是問題是如果我們在User中有很多個對象,那么clone方法就寫的很長,而且如果后面有修改,在User中新增屬性,這個地方也要改。

那么,有沒有什么辦法可以不需要修改,一勞永逸呢?

2、序列化實現(xiàn)深拷貝

我們可以借助序列化來實現(xiàn)深拷貝。先把對象序列化成流,再從流中反序列化成對象,這樣就一定是新的對象了。

序列化的方式有很多,比如我們可以使用各種JSON工具,把對象序列化成JSON字符串,然后再從字符串中反序列化成對象。

如使用fastjson實現(xiàn):

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

也可實現(xiàn)深拷貝。

除此之外,還可以使用Apache Commons Lang中提供的SerializationUtils工具實現(xiàn)。

我們需要修改下上面的User和Address類,使他們實現(xiàn)Serializable接口,否則是無法進行序列化的。

class User implements Serializable class Address implements Serializable

然后在需要拷貝的時候:

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

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

總結(jié)

當我們使用各類BeanUtils的時候,一定要注意是淺拷貝還是深拷貝,淺拷貝的結(jié)果就是兩個對象中的引用對象都是同一個地址,只要發(fā)生改變,都會有影響。

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

好了,以上就是今天的全部內(nèi)容了。

責任編輯:haq

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

    關(guān)注

    30

    文章

    4956

    瀏覽量

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

    關(guān)注

    4

    文章

    317

    瀏覽量

    28811

原文標題:這樣的爛代碼,我實習(xí)的時候都寫不出來!

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

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

掃碼添加小助手

加入工程師交流群

    評論

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

    搞笑諾貝爾:科學(xué)界的邪修盛宴

    真正的智慧有時候就藏在不為人關(guān)注的日常里
    的頭像 發(fā)表于 11-21 22:17 ?3281次閱讀
    搞笑諾貝爾:科學(xué)界的邪修盛宴

    伺服電機和步進電機的核心區(qū)別

    大家在運動控制的時候是用伺服電機的還是用步進電機做定位控制了?為什么有時候用伺服電機,有時候又步進電機?這是什么原因呢?本期我們就一起來探討一下步進電機和伺服電機的區(qū)別!
    的頭像 發(fā)表于 10-15 14:32 ?1554次閱讀
    伺服電機和步進電機的核心區(qū)別

    H743的LPUART1接收數(shù)據(jù)有時候出錯怎么解決?

    MODBUS協(xié)議,從機回復(fù)的都是正確的。LPUART1接收到就有時不正確。 波特率9600,時鐘用的是內(nèi)部CSI。 同一塊兒板子有時出現(xiàn),有時正常。有的板子出現(xiàn),有的板子不出現(xiàn)。
    發(fā)表于 09-22 06:58

    某車聯(lián)網(wǎng)終端有時無法短信喚醒的故障分析

    某車企反饋車載物聯(lián)網(wǎng)終端4G下有時無法接收短信,導(dǎo)致車輛無法喚醒。查詢短信中心記錄,下發(fā)失敗原因有時是“用戶不在服務(wù)區(qū)”,有時是“用戶關(guān)機”。實際終端一直保持開機狀態(tài)。
    的頭像 發(fā)表于 07-14 10:39 ?639次閱讀
    某車聯(lián)網(wǎng)終端<b class='flag-5'>有時</b>無法短信喚醒的故障分析

    ++和--運算時數(shù)據(jù)在正數(shù)時計算是正確的,負數(shù)時有時候會溜掉計數(shù),為什么?

    我用STM32G4做的無刷驅(qū)動板,用TIM3的霍爾模式來讀取電機的霍爾值,在程序里我寫了一段計數(shù)程序,當用手旋轉(zhuǎn)電機正轉(zhuǎn)時每次霍爾值改變計數(shù)值做++運算,反轉(zhuǎn)時--運算,轉(zhuǎn)動慢的時候沒有任何問題
    發(fā)表于 06-23 07:25

    DC紋波測試問題

    為什么在測試DC紋波的時候,為什么有時候測出來是幾十mv,有時候測出來是20多V,是測試方法有問題嗎,還是別的注意事項,用的200M的帶寬
    發(fā)表于 06-08 22:00

    【經(jīng)驗分享】玩轉(zhuǎn)FPGA串口通信:從“幻覺調(diào)試”到代碼解析

    FPGA開發(fā),思路先行!玩FPGA板子,讀代碼是基本功!尤其對從C語言轉(zhuǎn)戰(zhàn)FPGA的“寶貝們”來說,適應(yīng)流水線(pipeline)編程可能需要點時間。上篇點燈代碼解讀了基礎(chǔ),而如果能親手寫出串口通訊
    的頭像 發(fā)表于 06-05 08:05 ?996次閱讀
    【經(jīng)驗分享】玩轉(zhuǎn)FPGA串口通信:從“幻覺調(diào)試”到<b class='flag-5'>代碼</b>解析

    CY7C68013A跑著跑著掉線是什么原因?qū)е碌模?/a>

    RT所示,我現(xiàn)在用這顆USB芯片,跑的USB High Speed模式,在win10上會出現(xiàn)跑著跑著USB設(shè)備消失了,設(shè)備管理器中都也找不到該設(shè)備了,必須要重新插拔一下才可以,有時候1天就會出現(xiàn)一次,有時候3-4天出現(xiàn)一次。這可能是什么原因造成的?
    發(fā)表于 06-03 07:56

    調(diào)用DLPC34XX_WriteOperatingModeSelect(DLPC34XX_OM_SENS_INTERNAL_PATTERN)函數(shù),返回的是1,為什么?

    拒絕API訪問,而且這是偶然的,我們大多情況調(diào)用這個函數(shù)是返回0的,但是有時候突然返回1,并且在一段時間內(nèi)一直返回1,我們不知道如何解決。然而,這種情況在32位代碼中不會出現(xiàn)。
    發(fā)表于 02-17 08:29

    使用ADCPRO軟件進行測試的時候,點擊左上角的“Acquire”按鍵后,有時數(shù)據(jù)采不上來,是什么原因?

    在使用ADCPRO軟件進行測試的時候,點擊左上角的“Acquire”按鍵后,有時數(shù)據(jù)采不上來,左上角的狀態(tài)欄一直閃爍,下面采集狀態(tài)顯示100%。有時候是好的,可以采集上來。請問是什么原因。。。
    發(fā)表于 02-14 06:31

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

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

    用ADS1298在16K采樣率下,SPI SCLK為18MHZ.每次中斷,讀走8個通道數(shù)據(jù),有時候后四個通道讀出均為0,為什么?

    我用ADS1298在16K采樣率下,SPI SCLK為18MHZ.每次中斷,讀走8個通道數(shù)據(jù),但是有時候后四個通道讀出均為0,重新復(fù)位就好了。但是通過示波器看卻是有SCLK發(fā)出,讀數(shù)據(jù)很不穩(wěn)定。
    發(fā)表于 02-11 08:31

    tvp5150+DSP采集的圖像在CCS上顯示為左半部分和右半部分顛倒,怎么回事?

    如圖所示,左右顛倒了,貌似場同步信號延遲了半行。但是有時候是顛倒的,有時候又是正常顯示的。
    發(fā)表于 02-07 08:31

    如何在日常開發(fā)過程中提高代碼質(zhì)量

    。 提高代碼質(zhì)量是一個系統(tǒng)工程,本文主要介紹開發(fā)人員如何在日常開發(fā)過程中提高代碼質(zhì)量。 01 什么是代碼質(zhì)量? 代碼質(zhì)量一般用于衡量代碼的“
    的頭像 發(fā)表于 01-23 09:09 ?1168次閱讀
    如何在日常開發(fā)過程中提高<b class='flag-5'>代碼</b>質(zhì)量

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

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