日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

使用各类BeanUtils的时候,切记注意这个坑!

發(fā)布時間:2025/3/16 编程问答 22 豆豆
生活随笔 收集整理的這篇文章主要介紹了 使用各类BeanUtils的时候,切记注意这个坑! 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

在日常開發(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,我在《丟棄掉那些BeanUtils工具類吧,MapStruct真香!!!》中介紹過原因。這里就不再贅述了。

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

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

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

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

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

問題

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

我們的業(yè)務邏輯中,需要對訂單信息進行修改,在更改時,不僅要更新訂單的上面的屬性信息,還需要創(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)建一個新的對象,并復制其內(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 HomeAddress address;//省略構(gòu)造函數(shù)和setter/getter }

然后寫一段測試代碼:

User user = new User("Hollis", "hollischuang"); user.setAddress(new HomeAddress("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),原來的對象也受到了修改的影響。

這就是所謂的淺拷貝!

如何進行深拷貝

發(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@Overridepublic Object clone() throws CloneNotSupportedException {return super.clone();} }class User implements Cloneable{private String name;private String password;private HomeAddress address;//省略構(gòu)造函數(shù)和setter/getter@Overrideprotected Object clone() throws CloneNotSupportedException {User user = (User)super.clone();user.setAddress((HomeAddress)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的时候,切记注意这个坑!的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。