淺談Java中的深克隆和淺克隆(阿裡面試)

来源:https://www.cnblogs.com/liqiangchn/archive/2018/08/12/9465186.html
-Advertisement-
Play Games

在最近的秋招中,阿裡和多益網路都問到了這個問題,雖然很簡單,但是我還是想總結一下,感興趣的可以看一下我的 "個人博客網站(Spring+MyBatis+redis+nginx+mysql)" (適合菜鳥),最近會抽空把最近面試遇到的問題總結一下。 本文針對問題:深克隆和淺克隆的區別和實現方式?(阿裡 ...


在最近的秋招中,阿裡和多益網路都問到了這個問題,雖然很簡單,但是我還是想總結一下,感興趣的可以看一下我的個人博客網站(Spring+MyBatis+redis+nginx+mysql)(適合菜鳥),最近會抽空把最近面試遇到的問題總結一下。

本文針對問題:深克隆和淺克隆的區別和實現方式?(阿裡電面,多益網路的選擇題)

Talk is cheap

最近不止一次遇見深淺克隆(深複製,淺複製)的問題,除了印象中有個clone方法外一臉懵逼!!!克隆(複製)在Java中是一種常見的操作,目的是快速獲取一個對象副本。克隆分為深克隆和淺克隆。

淺克隆:創建一個新對象,新對象的屬性和原來對象完全相同,對於非基本類型屬性,仍指向原有屬性所指向的對象的記憶體地址。

深克隆:創建一個新對象,屬性中引用的其他對象也會被克隆,不再指向原有對象地址。

總之深淺克隆都會在堆中新分配一塊區域,區別在於對象屬性引用的對象是否需要進行克隆(遞歸性的)。

Show you my picture

pos:當前對象的地址;

son:son屬性所指向的地址;

name:對象的name屬性。

Show you my code

case1:

public class Son implements Serializable , Cloneable{
    private String name;
    private Son son;
    public Son() {
        super();
    }
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Son getSon() {
        return son;
    }

    public void setSon(Son son) {
        this.son = son;
    }

    @Override
    public String toString() {
        return super.toString();
    }
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

測試

public static void main(String[] args) throws Exception{
    // 創建父親(LiLiu),兒子(LiWu),孫子(LiLiu)並關聯
    Son father = new Son();
    father.setName("LiSi");
    Son son = new Son();
    son.setName("LiWu");
    Son grandSon = new Son();
    grandSon.setName("LiLiu");
    father.setSon(son);
    son.setSon(grandSon);
    // 調用clone方法
    Son fatherCopy =  (Son) father.clone();
    boolean flag1 = fatherCopy==father;
    boolean flag2 = fatherCopy.getSon() == son;
    boolean flag3 = fatherCopy.getSon().getSon() == grandSon;
    // 比較克隆後的地址
    System.out.println(flag1);// false
    System.out.println(flag2);// true
    System.out.println(flag3);// true
    // 比較Name
    flag1= fatherCopy.getName()==father.getName();
    flag2 = fatherCopy.getSon().getName() == son.getName();
    flag3 = fatherCopy.getSon().getSon().getName() == grandSon.getName();
    System.out.println(flag1);// true
    System.out.println(flag2);// true
    System.out.println(flag3);// true
    
    //將對象寫到流里    
    ByteArrayOutputStream byteOut=new ByteArrayOutputStream();    
    ObjectOutputStream objOut=new ObjectOutputStream(byteOut);    
    objOut.writeObject(father);
    //從流里讀出來    
    ByteArrayInputStream byteIn=new ByteArrayInputStream(byteOut.toByteArray());    
    ObjectInputStream objInput=new ObjectInputStream(byteIn);
    fatherCopy = (Son) objInput.readObject();
    flag1= fatherCopy==father;
    flag2 = fatherCopy.getSon() == son;
    flag3 = fatherCopy.getSon().getSon() == grandSon;
    System.out.println(flag1);// false
    System.out.println(flag2);// false
    System.out.println(flag3);// false
    
    // 比較Name
    flag1= fatherCopy.getName()==father.getName();
    flag2 = fatherCopy.getSon().getName() == son.getName();
    flag3 = fatherCopy.getSon().getSon().getName() == grandSon.getName();
    System.out.println(flag1);// false
    System.out.println(flag2);// false
    System.out.println(flag3);// false
}

從上文代碼及運行結果不難看出,如果對象實現Cloneable並重寫clone方法不進行任何操作時,調用clone是進行的淺克隆。而使用對象流將對象寫入流然後再讀出是進行的深克隆。

思考:既然實現Cloneable介面並重寫clone介面只能進行淺克隆。但是如果類的引用類型屬性(以及屬性的引用類型屬性)都進行淺克隆,直到沒有引用類型屬性或者引用類型屬性為null時,整體上就形成了深克隆。既對象的引用類型屬性和屬性的應用類型屬性都實現Coloneable,重寫clone方法併在clone方法中進行調用。

protected Object clone() throws CloneNotSupportedException {
    if (son != null) {
        this.son = (Son) son.clone();
    }
    return super.clone();
}

值得一提的是String比較特殊,它既不是基本數據類型,也沒有實現Cloneable介面,所以採用此種方法複製後仍然不能進行深複製。我試圖嘗試採用下文方法進行賦值,但是輸出結果仍然是和淺複製相同。(困惑,求解答

 protected Object clone() throws CloneNotSupportedException {
    if (son != null) {
        this.son = (Son) son.clone();
    }
     if (string !=null ){
         this.name = new String(name);
     }
    return super.clone();
}
 

但是這似乎並不影響,因為String是final類型的,每次修改都是創建一個新的對象,所以修改克隆的副本對象並不會影響原對象。

說幾句廢話

個人認為,在選擇深克隆方法時,應根據對象的複雜成都,如引用類型屬性是否有多層引用類型屬性關係。如果對象只有一層或者兩層引用類型的屬性,選擇思考中所提到的方法較為方便,反之則使用對象流。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 最終效果:瀏覽器地址欄輸入www.baidu.com訪問時,會顯示自己的網頁 1、創建文件 任意盤新建一個www.baidu.com文件,在該文件夾下新建WEB-INF文件、自己寫的一個html文件,一張圖片,然後在WEB-INF下新建一個classes文件、lib文件以及一個web.xml文件,在 ...
  • 給定兩個有序整數數組 nums1 和 nums2,將 nums2 合併到 nums1 中,使得 num1 成為一個有序數組。 說明: 初始化 nums1 和 nums2 的元素數量分別為 m 和 n。 你可以假設 nums1 有足夠的空間(空間大小大於或等於 m + n)來保存 nums2 中的元素 ...
  • 又有時間寫東西了,最近深感世事並不以人的美好願望而改變,還是要以積極地心態來適應新變化,多多關心身邊的人。 圖釘畫中一個圖釘代表一個像素,所以關鍵在於像素渣化,降低解析度,圖釘的色彩有限,還需要降低圖片的色彩數量,統計各種色彩的數量及位置。 以上都可以用Pillow完成,Pillow是Python中 ...
  • 網路上兩台主機的交互 ①根據IP找到對方主機 ②數據發送到對方指定的應用程式上,為了表示這些應用程式,引入了埠的概念。 常用埠: wed埠80 MySQL埠3306 有效埠 0~65535 ③定義通信規則,稱為協議。國際組織定義了通用協議TCP/IP協議 本地迴環地址:127.0.0.1( ...
  • 恢復內容開始 python爬蟲學習從0開始 第一次學習了python語法,迫不及待的來開始python的項目。首先接觸了爬蟲,是一個簡單爬蟲。個人感覺python非常簡潔,相比起java或其他面向對象的編程語言,動態語言不需要聲明函數或變數類型。python有20年的發展歷史,以簡潔高效聞名,pyt ...
  • 說概率前複習下歷史函數create_rand_list() #創建一個含有指定數量元素的listsum_fun() #累加len_fun() #統計個數multiply_fun() #累乘sum_mean_fun() #算數平均數sum_mean_rate() #算數平均數計算回報median_fu ...
  • 一、為什麼要用synchronized關鍵字 首先多線程中多個線程運行面臨共用數據同步的問題。 多線程正常使用共用數據時需要經過以下步驟: 1.線程A從共用數據區中複製出數據副本,然後處理。 2.線程A將處理好的數據副本寫入共用數據區。 3.線程B從共用數據區中複製出數據副本。 如此迴圈,直到線程結 ...
  • 最近維護一個項目,裡面用到ClientDataSet,由於之前接觸ClientDataSet比較少,所以這個星期補了一下關於ClientDataSet的知識,併在此記錄下我所瞭解到的並應用到實際項目中的ClientDataSet的知識。 項目新需求:1.從別的資料庫導入物料資料,並允許操作員做修改後 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...