该篇博客主要讨论在Objective-C中对于NSString变量的引用计数的不同处理。
为了在测试中打印方便,我们可以简单的定义一个宏:
#define MDMLog(str) ({NSString *name = @#str; NSLog(@"%@-->%@ %p, %zd", name, [str class], str, [str retainCount]);})
以下内容的测试环境为:Xcode9.4,SDK:iOS11.4,MRC
要想获取一个NSString对象,需要进行初始化NSString,所以我们就从NSString的初始化方法入手,进行NSString变量引用计数的分析。
字面量初始化
在对NSString初始化的方式中,字面量初始化是最直接的方法,我们来对字面量初始化的NSString对象进行引用计数查看:
NSString *str1 = @"123456789";
MDMLog(str1);
此时的打印结果为:
str1-->__NSCFConstantString 0x10beab060, -1
我们可以看到,此时生成的对象是NSString类簇中的__NSCFConstantString类的对象,同时由对象所处位置可以看出,此时生成的对象位于程序的数据区域,为可以一直存在的对象,所以该对象的计数为-1(即整数最大值),至于为何要在数据区域保存字符串常量,猜测是因为为了避免使用相同字符串常量造成的多次分配内存。该猜测可以由以下代码证实:
NSString *str1 = @"123456789";
MDMLog(str1);
NSString *str2 = @"123456789";
MDMLog(str2);
NSString *str3 = @"123456789";
MDMLog(str3);
打印结果如下:
str1-->__NSCFConstantString 0x107075060, -1
str2-->__NSCFConstantString 0x107075060, -1
str3-->__NSCFConstantString 0x107075060, -1
由此可见,在使用字符串变量进行初始化的NSString变量,它会指向数据区域的一块地址,并且该地址的计数为整数最大值。
便捷初始化
NSString的便捷初始化方法分为以下两种:
+ stringWithString:
该方法也是使用一个字符串常量来进行初始化的,按照字面量初始化的设计思路,该处生成的对象也应该是__NSCFConstantString类型的对象,我们来验证一下:
NSString *str1 = [NSString stringWithString:@"123456789"];
MDMLog(str1);
NSString *str2 = @"123456789";
MDMLog(str2);
NSString *str3 = @"123