在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

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

3天內不再提示

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

深圳東裕光大 ? 來源:Hollis ? 作者:Hollis ? 2021-08-27 10:23 ? 次閱讀

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

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

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

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

如我們經常在代碼中會對一個數據結構封裝成DO、SDO、DTO、VO等,而這些Bean中的大部分屬性都是一樣的,所以使用屬性拷貝類工具可以幫助我們節省大量的set和get操作。

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

1、Spring BeanUtils

2、Cglib BeanCopier

3、Apache BeanUtils

4、Apache PropertyUtils

5、Dozer

6、MapStucts

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

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

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

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

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

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

問題

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

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

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

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

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

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

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

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

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

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

淺拷貝?深拷貝?

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

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

2、深拷貝:對基本數據類型進行值傳遞,對引用數據類型,創建一個新的對象,并復制其內容,此為深拷貝。

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

先來定義兩個類:

public class Address { private String province; private String city; private String area; //省略構造函數和setter/getter } class User { private String name; private String password; private Address address; //省略構造函數和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());

以上代碼輸出結果為: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));

輸出結果:

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

可以發現,原來的對象也受到了修改的影響。

這就是所謂的淺拷貝!

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

如何進行深拷貝

發現問題之后,我們就要想辦法解決,那么如何實現深拷貝呢?

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

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

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

將上述代碼修改下,重寫clone方法:

public class Address implements Cloneable{ private String province; private String city; private String area; //省略構造函數和setter/getter@Override public Object clone() throws CloneNotSupportedException { return super.clone();

} } class User implements Cloneable{ private String name; private String password; private Address address; //省略構造函數和setter/getter @Override protected Object clone() throws CloneNotSupportedException { User user = (User)super.clone();

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

之后,在執行一下上面的測試代碼,就可以發現,這時候newUser中的address對象就是一個新的對象了。

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

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

2、序列化實現深拷貝

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

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

如使用fastjson實現:

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

也可實現深拷貝。

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

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

class User implements Serializable class Address implements Serializable

然后在需要拷貝的時候:

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

同樣,也可以實現深拷貝啦~!

總結

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

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

好了,以上就是今天的全部內容了。

責任編輯:haq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 代碼
    +關注

    關注

    30

    文章

    4829

    瀏覽量

    69068
  • 工具
    +關注

    關注

    4

    文章

    314

    瀏覽量

    27914

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

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

收藏 人收藏

    評論

    相關推薦

    ADS1191采集內部測試信TEST1Hz方波,有時候hen正常有時候不正常,是哪里出了問題?

    我采集內部測試信TEST1Hz方波,有時候hen正常 有時候不正常,不知道是哪里出現問題 很明顯第一幅圖是比較正常的 第二幅會出現問題。有沒有大神知道這個問題是什么原因 這是我
    發表于 12-30 06:30

    ADS1282讀寫寄存器讀不出正確的結果,有時候是全0,為什么?

    的什么值,再回讀寄存器內容的時候就讀不出正確的結果,有時候是全0,有時候是前幾個寄存器值對而后邊的不對,還有完全不對的情況。讀采集數據也是無規律的亂變的數值或者全0或者滿量程值等錯誤數據,與實際輸入
    發表于 12-13 06:15

    DAC8563xa中的10腳原本是輸出2.5v電壓,現在有時候有輸出2.5v,有時候沒有,為什么?

    DAC8563xa中的10腳原本是輸出2.5v電壓,現在有時候有輸出2.5v,有時候沒有,是不是基本上能確定是芯片壞了
    發表于 11-28 07:22

    LMX2595RHAR有時候鎖不住頻怎么回事?

    LMX2595RHAR 有時候鎖不住頻怎么回事? 6G的信號有時能鎖住,有時又鎖不住,但是我們用的另外一個頻點,4.5G,又沒這個現象 鎖不住的時候就是上圖那樣的,它的VT信號也是一
    發表于 11-11 06:38

    TAS5711有時候出現PLL autolock err,為什么?

    TAS5711有時候出現PLL autolock err
    發表于 10-30 06:02

    TAS5719調試DEMO板,有時候輸出幅度會不一樣,為什么?

    我想了解TI的TAS5719的壓限功能。并且我發現調試DEMO板,有時候輸出幅度會不一樣!比如,第一次開機我可以輸出電平可以壓在4V左右,但我僅僅調試音量,有可以輸出6V-7V.輸出信號波形失真了!
    發表于 10-15 08:19

    運放單電源供電,示波器直流耦合檔測試,為什么有時候示波器測的輸出信號出現負值?

    運放單電源供電,示波器直流耦合檔測試,為什么有時候示波器測的輸出信號出現負值?請解答,謝謝!
    發表于 08-29 08:25

    LTM8053,-12V輸出,連續開關機,有時候啟動不了是什么原因?

    LTM 8053,-12V輸出,連續開關機,有時候啟動不了,是什么原因?
    發表于 07-24 07:30

    ESP8266利用IO中斷接收紅外,有時候會出現波形少了的現象怎么解決?

    抓取紅外接收管的波形,與IO翻轉的波形,有時候會出現波形少了的現象,即是紅外接收管的信號有些沒有觸發到IO進入中斷。 官方的demo,基于RTOS的,只有發射代碼,沒有接收。 http://bbs.espressif.com/viewtopic.php?f=31&
    發表于 07-09 06:17

    esp32使用了nvs flash,有時候寫nvs容易使模塊復位,為什么?怎么解決?

    你好,我在使用該產品的時候, 需要一些數據能夠掉電記憶功能,使用了nvs flash,在有時候寫nvs的時候,容易使模塊復位? 不知道是什么原因,有解決的辦法嗎?
    發表于 06-26 06:43

    模組有時候復位重啟后輸出日志為“REBOOT_CAUSE_SECURITY_PMU_POWER_ON_RESET”的原因?

    情況:模組有時候復位重啟后輸出日志為“REBOOT_CAUSE_SECURITY_PMU_POWER_ON_RESET”。
    發表于 06-04 07:53

    IAR for stm8編譯很慢很慢有時候就無響應了是怎么回事?

    小弟第一次用這個編譯器,IAR for stm8編譯很慢很慢有時候就無響應了,有遇到過得嗎?
    發表于 05-08 06:00

    STM32F767ZI-Nucleo開發板跑官方HTTP例程有時候不能自動獲取IP,為什么?

    STM32F767ZI-Nucleo開發板跑官方HTTP例程有時候不能自動獲取IP,調試中發現只要是不能獲取IP時很大幾率就是以太網中斷不能進去。想請問一下這可能是什么問題引起的,唯一的改動就是
    發表于 04-25 06:02

    使用lcd顯示攝像頭數據的時候有時候會出現撕裂現象怎么解決?

    ic:h750 平臺:kile 問題描述:使用lcd顯示攝像頭數據的時候有時候會出現撕裂現象,且沒有規律,有時候很久出現一次,有時候經常出現;工程使用了freertos,有四個任務,
    發表于 04-07 08:05

    如何寫出時序最優的HDL代碼?如何寫出時序裕量足夠的代碼

    你想寫出可以跑出700M以上的代碼嗎,直逼FPGA內部PLL的極限。
    的頭像 發表于 03-12 09:59 ?988次閱讀
    如何<b class='flag-5'>寫出</b>時序最優的HDL<b class='flag-5'>代碼</b>?如何<b class='flag-5'>寫出</b>時序裕量足夠的<b class='flag-5'>代碼</b>?
    主站蜘蛛池模板: 亚洲理论视频 | 久久99精品国产麻豆宅宅 | 在线高清视频大全 | 国产农村妇女毛片精品久久久 | 久久久伊香蕉网站 | 看天堂 | 九九热视频免费在线观看 | 干美女在线视频 | 黑人黑粗硬视频 | 欧美特黄特色aaa大片免费看 | 色视频网 | 中文天堂在线观看 | 夜夜骚视频 | 亚洲综合精品一区二区三区中文 | www.毛片网站 | 天天爽天天爽天天片a久久网 | 欧美性free免费 | 国产小视频网站 | 天天干夜干 | 国模巴| 黄色在线播放网址 | 九九热在线免费视频 | 777欧美 | 亚洲国产高清人在线 | 99热这里只有精品一区二区三区 | 在线色色视频 | 米奇影院777 | 国产精品久久新婚兰兰 | 最新色视频 | wwwxx在线| 日本不卡1 | 很黄很暴力 很污秽的小说 很黄很黄叫声床戏免费视频 | 亚洲黄网址 | 中文字幕二区三区 | 亚洲韩国欧美一区二区三区 | 午夜三级视频 | 久久青草国产精品一区 | 日本在线不卡免 | 久久性 | 天天操夜夜操 | 激情综合五月亚洲婷婷 |