NSString retainCount探究

一、從一個(gè)引用計(jì)數(shù)的問題說起

題目詳情:下面代碼輸出結(jié)果

NSString *string = [[NSString alloc]initWithString:@"hello world"];
NSLog(@"stringRetainCount:%ld",[string retainCount]);

想當(dāng)然以為結(jié)果是1,但是實(shí)際結(jié)果是-1

 YQSourceCodeTest[916:12587] stringRetainCount:-1

二、然后進(jìn)一步探索,發(fā)現(xiàn)有如下發(fā)現(xiàn):

NSString *str = [[NSString alloc]initWithFormat:@"123456789"];
NSLog(@"initWithFormat(%@)-stringClass(%@)-retainCount(%ld)",str,[str class],[str retainCount]);
        
NSString *str1 = [[NSString alloc]initWithFormat:@"1234567890"];
NSLog(@"initWithFormat(%@)-stringClass(%@)-retainCount:%ld",str1,[str1 class],[str1 retainCount]);
        
NSString *str2 = [[NSString alloc]initWithFormat:@"文"];
NSLog(@"initWithFormat(%@)-stringClass(%@)-retainCount:%ld",str2,[str2 class],[str2 retainCount]);
        
NSString *str3 = @"123456789";
NSLog(@"(%@)-stringClass(%@)-retainCount:%ld",str3,[str3 class],[str3 retainCount]);
        
NSString *str4 = @"1234567890";
NSLog(@"(%@)-stringClass(%@)-retainCount:%ld",str4,[str4 class],[str4 retainCount]);
        
NSString *str5 = [[NSString alloc]initWithString:@"123456789"];
NSLog(@"initWithString(%@)-stringClass(%@)-retainCount:%ld",str5,[str5 class],[str5 retainCount]);
        
NSString *str6 = [[NSString alloc]initWithString:@"1234567890"];
NSLog(@"initWithString(%@)-stringClass(%@)-retainCount:%ld",str6,[str6 class],[str6 retainCount]);

具體輸出結(jié)果如下:

YQSourceCodeTest[1206:33010] initWithFormat(123456789)-stringClass(NSTaggedPointerString)-retainCount(-1)
YQSourceCodeTest[1206:33010] initWithFormat(1234567890)-stringClass(__NSCFString)-retainCount:1
YQSourceCodeTest[1206:33010] initWithFormat(文)-stringClass(__NSCFString)-retainCount:1
YQSourceCodeTest[1206:33010] (123456789)-stringClass(__NSCFConstantString)-retainCount:-1
YQSourceCodeTest[1206:33010] (1234567890)-stringClass(__NSCFConstantString)-retainCount:-1
YQSourceCodeTest[1206:33010] initWithString(123456789)-stringClass(__NSCFConstantString)-retainCount:-1
YQSourceCodeTest[1206:33010] initWithString(1234567890)-stringClass(__NSCFConstantString)-retainCount:-1

三、輸出結(jié)果探究

1、class類型共有三種:

  • NSTaggedPointerString :類型為標(biāo)記指針(Tagged Pointer:是一種為了提高iOS和macOS性能而設(shè)計(jì)的內(nèi)存優(yōu)化技術(shù)),
  • __NSCFString :運(yùn)行時(shí)創(chuàng)建的一種 NSString 子類,數(shù)據(jù)存儲(chǔ)在堆上
  • __NSCFConstantString:字符串常量,數(shù)據(jù)存儲(chǔ)在字符串常量區(qū)

2、retainCount

  • initWithFormat:字符串小于等于9長(zhǎng)度,retainCount = -1
  • initWithFormat:字符串大于9/為中文,retainCount = 1
  • 其他方法不受以上限制,始終retainCount = -1

四、其他延伸探究

1、編譯器內(nèi)存自動(dòng)優(yōu)化

NSString *str = @"abc";
NSLog(@"str:%p",str);
NSString *str1 = @"abc";
NSLog(@"str1:%p",str1);
NSString *str2 = [[NSString alloc]initWithString:@"abc"];
NSLog(@"str2:%p",str2);
NSString *str3 = [[NSString alloc]initWithFormat:@"abc"];
NSLog(@"str3:%p",str3);

前三個(gè)地址相同,輸出結(jié)果如下:

YQSourceCodeTest[1559:52231] str:0x100004058
YQSourceCodeTest[1559:52231] str1:0x100004058
YQSourceCodeTest[1559:52231] str2:0x100004058
YQSourceCodeTest[1559:52231] str3:0xbe8e4a5ff910c311

2、copy 和 mutableCopy 對(duì) retainCount影響

NSString *str = @"abc";
NSLog(@"str-retainCount:%ld",[str retainCount]);
NSString *str1 = [str copy];
NSLog(@"str-retainCount:%ld ; str1-retainCount:%ld",[str retainCount],[str1 retainCount]);
NSMutableString *strM = [str mutableCopy];
NSLog(@"str-retainCount:%ld ; strM-retainCount:%ld",[str retainCount],[strM retainCount]);
        
NSString *str2 = [[NSString alloc]initWithFormat:@"中文"];
NSLog(@"str2-retainCount:%ld",[str2 retainCount]);
NSString *str3 = [str2 copy];
NSLog(@"str2-retainCount:%ld ; str3-retainCount:%ld",[str2 retainCount],[str3 retainCount]);
NSMutableString *strM1 = [str2 mutableCopy];
NSLog(@"str2-retainCount:%ld ; strM1-retainCount:%ld",[str2 retainCount],[strM1 retainCount]);

針對(duì)NSTaggedPointerString 或者 __NSCFConstantString類型,copy和mutableCopy并不會(huì)增加引用技術(shù),__NSCFString類型下,copy使引用計(jì)數(shù)加1,具體結(jié)果如下:

YQSourceCodeTest[1664:58763] str-retainCount:-1
YQSourceCodeTest[1664:58763] str-retainCount:-1 ; str1-retainCount:-1
YQSourceCodeTest[1664:58763] str-retainCount:-1 ; strM-retainCount:1
YQSourceCodeTest[1664:58763] str2-retainCount:1
YQSourceCodeTest[1664:58763] str2-retainCount:2 ; str3-retainCount:2
YQSourceCodeTest[1664:58763] str2-retainCount:2 ; strM1-retainCount:1
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容