java?ArrayList的深拷貝與淺拷貝問題
一、前言
ArrayList
是我們經(jīng)常會(huì)用到的集合類,有時(shí)候我們?yōu)榱艘桓淖冊(cè)瓉淼臄?shù)據(jù)需要重新拷貝一個(gè)新的ArrayList
,今天在使用ArrayList
拷貝時(shí)遇到了一些問題,這里整理并記錄一下。
二、準(zhǔn)備
首先: ArrayList
的常見的拷貝方法有很多,其中都是淺拷貝
這里介紹幾種淺拷貝的方式:
1.通過構(gòu)造函數(shù)方法拷貝:
List<Integer> newList = new ArrayList<>(list);
2.addAll()
方法
List<Integer> newList = new ArrayList<>(); newList.addAll(list);
3.Collections.copy
方法
List<Integer> newList = new ArrayList<>(); newList.addAll(list); Collections.copy(newList, list)
4.stream
方法
java 8 的新特性
List<Integer> newList = list.stream().collect(toList());
另外一點(diǎn)
clone()
方式有些特殊,最開始我以為通過clone()
是實(shí)現(xiàn)深拷貝。
但其實(shí)clone()也是淺拷貝,原因如下:
因?yàn)橥ǔN覀兪褂玫念愋褪?code>Interger或者String
類型的List,Interger和String
類型都是不可變類,那么只需要通過淺拷貝拷貝一層即可。
給人的感覺是完全重新生成了一個(gè)新的ArrayList。
但是如果我們將類型改成我們自己的類型時(shí),就會(huì)出問題。
三、測(cè)試
將類型改成對(duì)象,在試一下:
// 模擬些數(shù)據(jù) Shard shard1 = new Shard(1,"張三","node1"); Shard shard2 = new Shard(2,"李四","node2"); Shard shard3 = new Shard(3,"王五","node3"); List<Shard> list = Arrays.asList(shard1, shard2, shard3); // 拷貝一個(gè)新的list List<Shard> newList = new ArrayList<>(); newList.addAll(list); Collections.copy(newList, list); // 修改新的list里數(shù)據(jù) newList.forEach(e -> e.setShardNum(4)); // 遍歷舊的list list.forEach(e -> System.out.println(e.getShardNum()));
結(jié)果:
4
4
4
可以看出這樣的List拷貝都是淺拷貝,都是拷貝的對(duì)象的引用,并沒有真正的去深拷貝。
大家可以試試別的方法,應(yīng)該都是不行的。
四、深拷貝
那么如何實(shí)現(xiàn)一個(gè)深拷貝,網(wǎng)上的推薦是使用序列化方法可以實(shí)現(xiàn)深拷貝。
代碼邏輯貼下:
public class CloneUtil { @SuppressWarnings("unchecked") public static <T extends Serializable> T clone(T obj){ T cloneObj = null; //寫入字節(jié)流 try { ByteArrayOutputStream out = new ByteArrayOutputStream(); ObjectOutputStream obs = new ObjectOutputStream(out); obs.writeObject(obj); obs.close(); //分配內(nèi)存,寫入原始對(duì)象,生成新對(duì)象 ByteArrayInputStream ios = new ByteArrayInputStream(out.toByteArray()); ObjectInputStream ois = new ObjectInputStream(ios); //返回生成的新對(duì)象 cloneObj = (T) ois.readObject(); ois.close(); }catch(IOException e){ e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); } return cloneObj; } }
使用:
for(Shard shard: list) { Shard newShard = CloneUtil.clone(shard); newList.add(newShard); }
注意點(diǎn):所有需要拷貝到的對(duì)象,通通要實(shí)現(xiàn)Serializable
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
多線程Thread,Runnable,Callable實(shí)現(xiàn)方式
這篇文章主要為大家詳細(xì)介紹了Java多線程如何實(shí)現(xiàn)Thread,Runnable,Callable的方式,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08SpringBoot Security前后端分離登錄驗(yàn)證的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot Security前后端分離登錄驗(yàn)證的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09最新log4j2遠(yuǎn)程代碼執(zhí)行漏洞(附解決方法)
Apache?Log4j2?遠(yuǎn)程代碼執(zhí)行漏洞攻擊代碼,該漏洞利用無需特殊配置,經(jīng)多方驗(yàn)證,Apache?Struts2、Apache?Solr、Apache?Druid、Apache?Flink等均受影響,本文就介紹一下解決方法2021-12-12Java java.lang.InstantiationException異常案例詳解
這篇文章主要介紹了Java java.lang.InstantiationException異常案例詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08基于Mybatis實(shí)現(xiàn)動(dòng)態(tài)數(shù)據(jù)源切換的示例代碼
在當(dāng)今的互聯(lián)網(wǎng)應(yīng)用中,微服務(wù)大行其道,隨著業(yè)務(wù)的發(fā)展和擴(kuò)展,單一的數(shù)據(jù)庫無法滿足日益增長(zhǎng)的數(shù)據(jù)需求,本文將基于 JDK17 + Spring Boot 3 和 MyBatis 框架實(shí)現(xiàn)動(dòng)態(tài)切換數(shù)據(jù)源功能,需要的朋友可以參考下2024-09-09Java?8函數(shù)式接口之Consumer用法示例詳解
這篇文章主要為大家介紹了Java?8函數(shù)式接口之Consumer用法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07linux下idea、pycharm等輸入中文拼音時(shí)滿3個(gè)字母后無法繼續(xù)拼音輸入的問題
這篇文章主要介紹了linux下idea、pycharm等輸入中文拼音時(shí)滿3個(gè)字母后無法繼續(xù)拼音輸入的問題,本文通過圖文并茂的形式給大家分享解決方法,需要的朋友可以參考下2021-04-04