NSString屬性何時用strong何時用copy?
前言
我們在聲明一個NSString屬性時,對于其內存相關特性,通常有兩種選擇(基于ARC環(huán)境):strong與copy。那這兩者有什么區(qū)別呢?什么時候該用strong,什么時候該用copy呢?讓我們先來看個例子。
代碼驗證
@property (nonatomic, strong) NSString *myStrongString; @property (nonatomic, copy) NSString *myCopyString; - (void)stringTest { NSMutableString *mutableStr = [NSMutableString stringWithFormat:@"https://"]; self.myStrongString = mutableStr; self.myCopyString = mutableStr; NSLog(@"mutableStr:%p,%p", mutableStr,&mutableStr); NSLog(@"strongString:%p,%p", _myStrongString, &_myStrongString); NSLog(@"copyString:%p,%p", _myCopyString, &_myCopyString); //---------------分割線--------------------- [mutableStr appendString:@"devthinking.com"]; NSLog(@"strongString:%p,%p", _myStrongString, &_myStrongString); NSLog(@"copyString:%p,%p", _myCopyString, &_myCopyString); }
打印日志如下:
2016-11-09 14:14:18.532 DemoCollectOC[92929:1731791] mutableStr:0x60800026fe00,0x7fff549c9be8 2016-11-09 14:14:18.532 DemoCollectOC[92929:1731791] strongString:0x60800026fe00,0x7ff095402308 2016-11-09 14:14:18.533 DemoCollectOC[92929:1731791] copyString:0x6080004234a0,0x7ff095402310 2016-11-09 14:14:18.533 DemoCollectOC[92929:1731791] strongString:0x60800026fe00,0x7ff095402308 2016-11-09 14:14:21.039 DemoCollectOC[92929:1731791] copyString:0x6080004234a0,0x7ff095402310
結論
1.myStrongString跟mutableStr的指向地址始終一樣,都為0x60800026fe00,myCopyString跟mutableStr指向的地址不同,即為深copy,新開辟了一份內存;
2.賦值時,當原始字符串是mutable String時,存在此差異,當mutableStr變化時,myStrongString會隨著改變,而myCopyString則不會。通常我們不想讓其隨著改變,故用copy屬性較多;如果我們想其隨著改變,則用strong。
3. 如果原始賦值字符串為string時,則用copy和strong屬性是一樣的。
變量存儲地址
- &取地址符,取出來的是變量的存儲地址,如myStrongString myCopyString是存在堆里的,地址以0x7ff09開頭,mutableStr為臨時變量,是存在棧里的,以0x7fff5開頭。
- 直接p打印出來的地址,則是存儲內容實際存在的地址,其實里面存儲的還是地址,詳細的請看下一節(jié),但是我們可以用這個地址區(qū)間來判斷存儲的區(qū)域。
附:x/3gx查看對象內存
查看基本數據類型的內存時,可直接查看,16進制轉化一下就可以,查看對象時,則不可以,這跟String的結構體是相關的:
以NSString為例
NSString *str = @"a";
先打印出地址:
(lldb) p str (__NSCFConstantString *) $0 = 0x0000000109b3aa30 @"a"
再用x/3gx命令查看內存:
(lldb) x/3gx 0x0000000109b3aa30 0x109b3aa30: 0x000000010bf3e348 0x00000000000007c8 0x109b3aa40: 0x0000000109b276db
再查看0x0000000109b276db中的地址即為字母a的Ascii碼實際存在地址:
(lldb) p (char*)0x0000000109b276db (char *) $2 = 0x0000000109b276db "a"
可以直接在變量上右鍵上打開view memory of “*str”, 這就打開memory查看,在Address里面輸上0x0000000109b276db,即可查看每個字節(jié)的內容。
總結
以上就是這篇文章的全部內容,希望本文的內容對各位iOS開發(fā)者們能有所幫助,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關文章
iOS開發(fā)tips-UINavigationBar的切換效果
這篇文章主要為大家詳細介紹了iOS開發(fā)tips-UINavigationBar的切換效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-11-11iOS overFullScreen與fullScreen區(qū)別分析
這篇文章主要介紹了iOS overFullScreen與fullScreen區(qū)別分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-11-11iOS常用算法之兩個有序數組合并(要求時間復雜度為0(n))
這篇文章主要介紹了iOS常用算法之兩個有序數組合并(要求時間復雜度為0(n)),實現思路是先將一個數組作為合并后的數組, 然后遍歷第二個數組的每項元素,需要的朋友可以參考下2019-07-07