引用计数的几个问题学习总结

关于 NSString 的疑问

NSString *str = @"Joy";NSLog(@"%lu",[str retainCount]);NSLog(@"地址:%p",str);
打印结果:

2016-06-15 10:22:17.084 OCTestProject[2592:327118] 计数:184467440737095516152016-06-15 10:22:17.085 OCTestProject[2592:327118] 地址:0x1045077d0
会发现引用计数是一个很大的值,为什么?这是一个放在常量区的字符串常量,返回的结果是UINT_MAX值

关于 release 之后仍然为1的疑问

Student *stu = [[Student alloc] init];NSLog(@"%lu",[stu retainCount]);[stu release];NSLog(@"%lu",[stu retainCount]);
打印结果:

2016-06-15 12:07:50.608 OCTestProject[3437:361531] 12016-06-15 12:07:50.609 OCTestProject[3437:361531] 1
有人告诉我,是autoreleasepool的原因!!明确的说,和autoreleasepool完全没关系。

向一个被回收的对象发送retaincount消息,输出结果不确定,如果这块内存被复用了,那么这里就会造成程序崩溃。最后一次release之后,系统知道这块内存要进行回收了,但是只是进行一个标记,并不会将retaincount减去1,也没必要这么做了。直接标记,可以减少一次内存操作,加速对对象的回收,何乐而不为

什么对象自动加入到 autoreleasepool中

虽然在程序入口,已经帮我们加上了 autoreleasepool,但是并不是说大括号内的所有

对象都会交给autoreleasepool来处理

第一种

当使用alloc/new/copy/mutableCopy开始的方法进行初始化时,会生成并持有对象(也就是不需要pool管理,系统会自动的帮他在合适位置release)。例如:

NSObject *stu = [[NSObject alloc] init];
那么对于其他情况,例如

id obj = [NSMutableArray array];
这种情况会自动将返回值的对象注册到autorealeasepool,代码等效于:

@autorealsepool{ id __autorealeasing obj = [NSMutableArray array];}
第二种

__weak修饰符只持有对象的弱引用,而在访问引用对象的过程中,该对象可能被废弃。那么如果把对象注册到autorealeasepool中,那么在@autorealeasepool块结束之前都能确保对象的存在。

id __weak obj1 = obj0;NSLog(@"class=%@",[obj1 class]);
对应的模拟源码为

id __weak obj1 = obj0;id __autorealeasing tmp = obj1;NSLog(@"class=%@",[tmp class]);
第三种

id的指针或对象的指针在没有显式指定时会被附加上__autorealeasing修饰符

  • (nullable instancetype)stringWithContentsOfURL:(NSURL *)url encoding:(NSStringEncoding)enc error:(NSError **)error;
    等价于

NSString *str = [NSString stringWithContentsOfURL: encoding: error:<#(NSError * _Nullable __autoreleasing * _Nullable)#>]

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容