java?ArrayList的深拷貝與淺拷貝問題
一、前言
ArrayList
是我們經(jīng)常會用到的集合類,有時候我們?yōu)榱艘桓淖冊瓉淼臄?shù)據(jù)需要重新拷貝一個新的ArrayList
,今天在使用ArrayList
拷貝時遇到了一些問題,這里整理并記錄一下。
二、準備
首先: ArrayList
的常見的拷貝方法有很多,其中都是淺拷貝
這里介紹幾種淺拷貝的方式:
1.通過構造函數(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());
另外一點
clone()
方式有些特殊,最開始我以為通過clone()
是實現(xiàn)深拷貝。
但其實clone()也是淺拷貝,原因如下:
因為通常我們使用的類型是Interger
或者String
類型的List,Interger和String
類型都是不可變類,那么只需要通過淺拷貝拷貝一層即可。
給人的感覺是完全重新生成了一個新的ArrayList。
但是如果我們將類型改成我們自己的類型時,就會出問題。
三、測試
將類型改成對象,在試一下:
// 模擬些數(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); // 拷貝一個新的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()));
結果:
4
4
4
可以看出這樣的List拷貝都是淺拷貝,都是拷貝的對象的引用,并沒有真正的去深拷貝。
大家可以試試別的方法,應該都是不行的。
四、深拷貝
那么如何實現(xiàn)一個深拷貝,網(wǎng)上的推薦是使用序列化方法可以實現(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)存,寫入原始對象,生成新對象 ByteArrayInputStream ios = new ByteArrayInputStream(out.toByteArray()); ObjectInputStream ois = new ObjectInputStream(ios); //返回生成的新對象 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); }
注意點:所有需要拷貝到的對象,通通要實現(xiàn)Serializable
總結
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
多線程Thread,Runnable,Callable實現(xiàn)方式
這篇文章主要為大家詳細介紹了Java多線程如何實現(xiàn)Thread,Runnable,Callable的方式,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-08-08SpringBoot Security前后端分離登錄驗證的實現(xiàn)
這篇文章主要介紹了SpringBoot Security前后端分離登錄驗證的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-09-09Java java.lang.InstantiationException異常案例詳解
這篇文章主要介紹了Java java.lang.InstantiationException異常案例詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-08-08基于Mybatis實現(xiàn)動態(tài)數(shù)據(jù)源切換的示例代碼
在當今的互聯(lián)網(wǎng)應用中,微服務大行其道,隨著業(yè)務的發(fā)展和擴展,單一的數(shù)據(jù)庫無法滿足日益增長的數(shù)據(jù)需求,本文將基于 JDK17 + Spring Boot 3 和 MyBatis 框架實現(xiàn)動態(tài)切換數(shù)據(jù)源功能,需要的朋友可以參考下2024-09-09Java?8函數(shù)式接口之Consumer用法示例詳解
這篇文章主要為大家介紹了Java?8函數(shù)式接口之Consumer用法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-07-07linux下idea、pycharm等輸入中文拼音時滿3個字母后無法繼續(xù)拼音輸入的問題
這篇文章主要介紹了linux下idea、pycharm等輸入中文拼音時滿3個字母后無法繼續(xù)拼音輸入的問題,本文通過圖文并茂的形式給大家分享解決方法,需要的朋友可以參考下2021-04-04