iOS 圖文並茂的帶你瞭解深拷貝與淺拷貝

来源:http://www.cnblogs.com/beckwang0912/archive/2017/07/20/7212075.html
-Advertisement-
Play Games

一、概念與總結 1、淺拷貝 淺拷貝就是對記憶體地址的複製,讓目標對象指針和源對象指向同一片記憶體空間,當記憶體銷毀的時候,指向這片記憶體的幾個指針需要重新定義才可以使用,要不然會成為野指針。 淺拷貝就是拷貝指向原來對象的指針,使原對象的引用計數+1,可以理解為創建了一個指向原對象的新指針而已,並沒有創建一個 ...


一、概念與總結

     1、淺拷貝

     淺拷貝就是對記憶體地址的複製,讓目標對象指針和源對象指向同一片記憶體空間,當記憶體銷毀的時候,指向這片記憶體的幾個指針需要重新定義才可以使用,要不然會成為野指針。

      淺拷貝就是拷貝指向原來對象的指針,使原對象的引用計數+1,可以理解為創建了一個指向原對象的新指針而已,並沒有創建一個全新的對象。

      2、深拷貝

      深拷貝是指拷貝對象的具體內容,而記憶體地址是自主分配的,拷貝結束之後,兩個對象雖然存的值是相同的,但是記憶體地址不一樣,兩個對象也互不影響,互不幹涉。

     深拷貝就是拷貝出和原來僅僅是值一樣,但是記憶體地址完全不一樣的新的對象,創建後和原對象沒有任何關係。

   

    3、總結:

    深拷貝就是內容拷貝,淺拷貝就是指針拷貝。本質區別在於:

  • 是否開啟新的記憶體地址
  • 是否影響記憶體地址的引用計數

 

 二、示例分析

      在iOS中深拷貝與淺拷貝要更加的複雜,涉及到容器與非容器、可變與不可變對象的copy與mutableCopy。下麵用示例逐一分析:

      1、非集合對象的copy與mutableCopy

        1.1 不可變對象NSString 

- (void) noMutableNSStringTest
{
    NSString *str1 = @"test001";
    
    NSMutableString *str2 = [str1 copy];
    //copy返回的是不可變對象,str2不能被修改,因此會發生崩潰
    //[str2 appendString:@"test"];
    
    NSMutableString *str3 = [str1 mutableCopy];
    [str3 appendString:@"modify"];
    
    NSLog(@"str1:%p - %@ \r\n",str1,str1);
    NSLog(@"str2:%p - %@ \r\n",str2,str2);
    NSLog(@"str3:%p - %@ \r\n",str3,str3);
}

          列印結果:

2017-07-20 18:02:10.642 beck.wang[1306:169414] str1:0x106abdbd0 - test001 
2017-07-20 18:02:10.643 beck.wang[1306:169414] str2:0x106abdbd0 - test001 
2017-07-20 18:02:10.643 beck.wang[1306:169414] str3:0x608000260940 - test001modify 

         分析:str1、str2地址相同並且與str3地址不同,NSString的copy是淺拷貝,且copy返回的對象是不可變對象;mutableCopy是深拷貝。

     

      1.2 可變對象NSMutableString

- (void) mutableNSStringTest
{
    NSMutableString *mstr1 = [NSMutableString stringWithString:@"test002"];
    
    NSMutableString *mstr2 = [mstr1 copy];
    //copy返回的是不可變對象,mstr2不能被修改,因此會發生崩潰
    //[str2 appendString:@"test"];
    
    NSMutableString *mstr3 = [mstr1 mutableCopy];
    [mstr3 appendString:@"modify"];
    
    NSLog(@"mstr1:%p - %@ \r\n",mstr1,mstr1);
    NSLog(@"mstr2:%p - %@ \r\n",mstr2,mstr2);
    NSLog(@"mstr3:%p - %@ \r\n",mstr3,mstr3);
}

        列印結果:

2017-07-20 18:14:35.789 beck.wang[1433:180881] mstr1:0x610000075e40 - test002 
2017-07-20 18:14:35.790 beck.wang[1433:180881] mstr2:0xa323030747365747 - test002 
2017-07-20 18:14:35.790 beck.wang[1433:180881] mstr3:0x610000074480 - test002modify 

        分析:mstr1、mstr2、mstr3 地址都不同,NSMutableString對象copy與mutableCopy都是深拷貝,且copy返回的對象是不可變對象。

 

    2、集合對象的copy與mutableCopy

       2.1 不可變對象NSArray

- (void) mutableNSArrayTest
{
    NSArray *arry1 = [[NSArray alloc] initWithObjects:@"value1", @"value2",nil];
    
    NSArray *arry2 = [arry1 copy];
    NSArray *arry3 = [arry1 mutableCopy];
    
    NSLog(@"arry1:%p - %@ \r\n",arry1,arry1);
    NSLog(@"arry2:%p - %@ \r\n",arry2,arry2);
    NSLog(@"arry3:%p - %@ \r\n",arry3,arry3);
}

       列印結果:

2017-07-20 18:33:53.707 beck.wang[1502:194476] arry1:0x60800003b480 - (
    value1,
    value2
) 
2017-07-20 18:33:53.708 beck.wang[1502:194476] arry2:0x60800003b480 - (
    value1,
    value2
) 
2017-07-20 18:33:53.708 beck.wang[1502:194476] arry3:0x60800004cd20 - (
    value1,
    value2
) 

      分析:arry1、arry2 地址一樣,arr3 地址不一樣,NSArray的copy是淺拷貝,且copy返回的對象是不可變對象;mutableCopy是深拷貝。

      

      2.2 可變對象NSMutableArray

- (void) NSMutableArrayTest
{
    NSMutableArray *marry1 = [[NSMutableArray alloc] initWithObjects:@"value1", @"value2",nil];
    
    NSMutableArray *marry2 = [marry1 copy];
    
    //copy返回的是不可變對象,marry2不能被修改,因此會崩潰
    //[marry2 addObject:@"value3"];
    
    NSMutableArray *marry3 = [marry1 mutableCopy];
    
    NSLog(@"marry1:%p - %@ \r\n",marry1,marry1);
    NSLog(@"marry2:%p - %@ \r\n",marry2,marry2);
    NSLog(@"marry3:%p - %@ \r\n",marry3,marry3);
}

        列印結果:

2017-07-20 18:55:43.243 beck.wang[1577:204641] marry1:0x600000048d60 - (
    value1,
    value2
) 
2017-07-20 18:55:43.244 beck.wang[1577:204641] marry2:0x600000026000 - (
    value1,
    value2
) 
2017-07-20 18:55:43.244 beck.wang[1577:204641] marry3:0x6000000494b0 - (
    value1,
    value2
) 

       分析:marry1、marry2、marr3 地址都不一樣,NSMutableArray對象copy與mutableCopy都是深拷貝,且copy返回的對象是不可變對象。

 

       特別註意的是:對於集合類的可變對象來說,深拷貝並非嚴格意義上的深複製,只能算是單層深複製,即雖然新開闢了記憶體地址,但是存放在記憶體上的值(也就是數組裡的元素仍然之鄉員數組元素值,並沒有另外複製一份),這就叫做單層深複製。

       舉例說明:

- (void)singleNSMutableArrayTest
{
    NSMutableArray *marry1 = [[NSMutableArray alloc] init];
    
    NSMutableString *mstr1 = [[NSMutableString alloc]initWithString:@"value1"];
    NSMutableString *mstr2 = [[NSMutableString alloc]initWithString:@"value2"];
    
    [marry1 addObject:mstr1];
    [marry1 addObject:mstr2];
    
    NSMutableArray *marry2 = [marry1 copy];
    NSMutableArray *marry3 = [marry1 mutableCopy];
    
    NSLog(@"marry1:%p - %@ \r\n",marry1,marry1);
    NSLog(@"marry2:%p - %@ \r\n",marry2,marry2);
    NSLog(@"marry3:%p - %@ \r\n",marry3,marry3);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marry1[0],marry1[1]);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marry2[0],marry2[1]);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marry3[0],marry3[1]);
    
    NSLog(@"\r\n------------------修改原值後------------------------\r\n");
    [mstr1 appendFormat:@"aaa"];
    
    NSLog(@"marry1:%p - %@ \r\n",marry1,marry1);
    NSLog(@"marry2:%p - %@ \r\n",marry2,marry2);
    NSLog(@"marry3:%p - %@ \r\n",marry3,marry3);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marry1[0],marry1[1]);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marry2[0],marry2[1]);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marry3[0],marry3[1]);
}

        列印結果:

2017-07-20 19:48:24.539 beck.wang[1750:230132] marry1:0x60800004ae00 - (
    value1,
    value2
) 
2017-07-20 19:48:24.539 beck.wang[1750:230132] marry2:0x608000023f00 - (
    value1,
    value2
) 
2017-07-20 19:48:24.539 beck.wang[1750:230132] marry3:0x60800004abc0 - (
    value1,
    value2
) 
2017-07-20 19:48:24.540 beck.wang[1750:230132] 數組元素地址:value1:0x60800006df40 - value2:0x60800006cb40 
2017-07-20 19:48:24.540 beck.wang[1750:230132] 數組元素地址:value1:0x60800006df40 - value2:0x60800006cb40 
2017-07-20 19:48:24.540 beck.wang[1750:230132] 數組元素地址:value1:0x60800006df40 - value2:0x60800006cb40 
2017-07-20 19:48:24.540 beck.wang[1750:230132] 
------------------修改原值後------------------------
2017-07-20 19:48:24.540 beck.wang[1750:230132] marry1:0x60800004ae00 - (
    value1aaa,
    value2
) 
2017-07-20 19:48:24.540 beck.wang[1750:230132] marry2:0x608000023f00 - (
    value1aaa,
    value2
) 
2017-07-20 19:48:24.540 beck.wang[1750:230132] marry3:0x60800004abc0 - (
    value1aaa,
    value2
) 
2017-07-20 19:48:24.541 beck.wang[1750:230132] 數組元素地址:value1:0x60800006df40 - value2:0x60800006cb40 
2017-07-20 19:48:24.541 beck.wang[1750:230132] 數組元素地址:value1:0x60800006df40 - value2:0x60800006cb40 
2017-07-20 19:48:24.541 beck.wang[1750:230132] 數組元素地址:value1:0x60800006df40 - value2:0x60800006cb40 

      分析:在修改原值之前,marry1、marry2、marr3 地址都不一樣,很明顯copy和mutableCopy都是深拷貝,但是從修改原值後的列印結果來看,這裡的深拷貝只是單層深拷貝:新開闢了記憶體地址,但是數組中的值還是指向原數組的,這樣才能在修改原值後,marry2 marr3中的值都修改了。另外,從列印的數組元素地址可以很明顯的看出來,修改前後marry1、marry、marr3的數組元素地址都是一模一樣的,更加佐證了這一點。

 

      2.3 思維擴展:集合對象的完全深拷貝

            2.2中提到了集合類的對象來說,深拷貝只是單層深拷貝,那有沒有辦法實現每一層都深拷貝呢?回答是肯定的,目前我們可以這麼做:

          (1)歸檔解檔大法

- (void) deplyFullCopy
{
    NSMutableArray *marry1 = [[NSMutableArray alloc] init];
    
    NSMutableString *mstr1 = [[NSMutableString alloc]initWithString:@"value1"];
    NSMutableString *mstr2 = [[NSMutableString alloc]initWithString:@"value2"];
    
    [marry1 addObject:mstr1];
    [marry1 addObject:mstr2];
    
    NSData *data = [NSKeyedArchiver archivedDataWithRootObject:marry1];
    NSArray *marray2 = [NSKeyedUnarchiver unarchiveTopLevelObjectWithData:data error:nil];
    
    NSLog(@"marry1:%p - %@ \r\n",marry1,marry1);
    NSLog(@"marry2:%p - %@ \r\n",marray2,marray2);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marry1[0],marry1[1]);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marray2[0],marray2[1]);
}

          列印結果:

2017-07-20 20:04:38.726 beck.wang[1833:242158] marry1:0x600000048a00 - (
    value1,
    value2
) 
2017-07-20 20:04:38.726 beck.wang[1833:242158] marry2:0x600000049780 - (
    value1,
    value2
) 
2017-07-20 20:04:38.726 beck.wang[1833:242158] 數組元素地址:value1:0x600000066300 - value2:0x600000067000 
2017-07-20 20:04:38.726 beck.wang[1833:242158] 數組元素地址:value1:0x600000066740 - value2:0x600000066f40 

        分析:我們可以看到,開闢了新的記憶體地址的同時,數組元素的指針地址也不同了,實現了完全的深拷貝。

      (2)- (instancetype)initWithArray:(NSArray<ObjectType> *)array copyItems:(BOOL)flag;

- (void) deplyFullCopy2
{
    NSMutableArray *marry1 = [[NSMutableArray alloc] init];
    
    NSMutableString *mstr1 = [[NSMutableString alloc]initWithString:@"value1"];
    NSMutableString *mstr2 = [[NSMutableString alloc]initWithString:@"value2"];
    
    [marry1 addObject:mstr1];
    [marry1 addObject:mstr2];
    
    NSArray *marray2 = [[NSArray alloc] initWithArray:marry1 copyItems:YES];
    
    NSLog(@"marry1:%p - %@ \r\n",marry1,marry1);
    NSLog(@"marry2:%p - %@ \r\n",marray2,marray2);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marry1[0],marry1[1]);
    NSLog(@"數組元素地址:value1:%p - value2:%p \r\n",marray2[0],marray2[1]);
}

        列印結果:

2017-07-20 20:08:04.201 beck.wang[1868:246161] marry1:0x610000050320 - (
    value1,
    value2
) 
2017-07-20 20:08:04.202 beck.wang[1868:246161] marry2:0x6100002214c0 - (
    value1,
    value2
) 
2017-07-20 20:08:04.202 beck.wang[1868:246161] 數組元素地址:value1:0x610000265600 - value2:0x610000266400 
2017-07-20 20:08:04.202 beck.wang[1868:246161] 數組元素地址:value1:0xa003165756c61766 - value2:0xa003265756c61766 

      分析:同上。

 

三、準則

      No1:可變對象的copy和mutableCopy方法都是深拷貝(區別完全深拷貝與單層深拷貝) 。

    No2:不可變對象的copy方法是淺拷貝,mutableCopy方法是深拷貝。

    No3:copy方法返回的對象都是不可變對象。

 

      萬語千言匯成一張圖

 

 

 深淺拷貝講完了,後續我會抽時間圖文並茂的分析記憶體定義語句的區別, 鑒於水平有限,如果有不當之處還望批評指正,謝謝!


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

-Advertisement-
Play Games
更多相關文章
  • 將數據存儲到文件中並讀取數據 1、新建FilePersistenceTest項目,並修改activity_main.xml中的代碼,如下:(只加入了EditText,用於輸入文本內容,不管輸入什麼按下back鍵就丟失,我們要做的是數據被回收之前,將它存儲在文件中) 2、修改MainActivity中 ...
  • 最近遇到了一個問題,就是在`UIWebView`的代理方法里,執行`document.title`的`js`代碼無法獲取網頁標題 ...
  • 今天來介紹一個小工具`ccache`,其可以提高`xcode`的編譯速度。說起緣由,是因為我的蘋果電腦配置比較低,而每次開發調試或測試打包都需要編譯工程,雖然項目工程代碼量不算大,但是編譯的時間還是很長,尤其是修改了頭文件或者`Archive`打包時,看著進度條像蝸牛在爬一樣,心裡都有小動物在奔騰.... ...
  • 在開發`iOS`應用程式的過程中,經常需要根據不同的需求,切換到不同的項目配置,或者打不同的包(測試環境、開發環境、生產環境等等),如果每次都是手動配置,一則比較麻煩,二則容易配置錯,那麼有沒有更好的方案來解決這個問題呢?答案是:有的。我們可以根據不同的需求,創建不同的`target`,在不同需求要... ...
  • Android精選源碼 android實現最簡潔的標簽(label/tag)選擇/展示控制項 懂得智能配色的ImageView,還能給自己設置多彩的陰影哦 NicePhoto-基於 Kotlin 開發的 一款超簡單的圖片瀏覽+設置壁紙... 你的桌面從未如此炫酷(一句代碼搞定) 仿支付寶首頁下拉刷新 ...
  • 在開發`iOS`的客戶端應用時,經常需要從伺服器下載圖片,雖然系統提供了下載工具:NSData、NSURLSession等等方法,但是考慮到圖片下載過程中,需要考慮的因素比較多,比如:非同步下載、圖片緩存、錯誤處理、編碼解碼等,以及實際需要中根據不同網路載入不同畫質的圖片等等需求,因此下載操作不是一個... ...
  • 一,效果圖。 二,代碼。 ViewController.h ViewController.m ...
  • 最近研究了一下項目的組件化,把`casa`、`bang`、`limboy`的有關組件化的博客看了一遍,學到了不少東西,對目前業界的組件化方案有了一定的瞭解。這些高質量的博客大致討論了組件化的三種方案:`url-block`、`protocol-class`(和`url-controller`類似)、... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...