博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
ARC对self的内存管理
阅读量:6477 次
发布时间:2019-06-23

本文共 1611 字,大约阅读时间需要 5 分钟。

来源:http://blog.sunnyxx.com/2015/01/17/self-in-arc/

记录下前两天的一次讨论,源于网络库YTKNetwork中的- start方法其中的几行代码:

- (void)start {

// ......

YTKRequest *strongSelf =self;

[strongSelf.delegate requestFinished:strongSelf];

if(strongSelf.successCompletionBlock) {

strongSelf.successCompletionBlock(strongSelf);

}

[strongSelf clearCompletionBlock];

}

看起来比较有违常理,所以和猿题库的@晨钰Lancy,@唐巧以及网易的@老汉一起讨论了下这个问题。

具体的问题大概是这样:

调用方(如view controller)实例化并强引用YTKRequest对象,将自己作为其delegate

调用方调用YTKRequest的- start方法发起网络请求

调用方在- requestFinished:中执行了self.request = nil;

YTKRequest中,- start方法在回调完- requestFinished:后BAD_ACCESS

也就是说,- start方法还未返回时,self就被外部释放了。作者发现了这个潜在的问题,所以在方法局部增设了一个strongSelf的强引用来保证self的生命周期延续到方法结束。问题是解决了,但是更希望知道原因。

简化说明就是:

- (void)foo {

// self被delegate持有

[self.delegate callout];// 外部释放了这个对象

// 这里self野指针

}

现在想想还是比较不符合常理,入参的self居然不能保证这个函数执行完成。后来查阅了下文档,发现是ARC的(gao)机(de)制(gui),clang的中有明确的解释,总结如下:

ARC下,self既不是strong也不是weak,而是unsafe_unretained的,也就是说,入参的self被表示为:(init系列方法的self除外)

- (void)start {

const__unsafe_unretained YTKRequest *self;

// ...

}

在方法调用时,ARC不会对self做retain或release,生命周期全由它的调用方来保证,如果调用方没有保证,就会出现上面的crash

ARC这样做的原因是性能优化,objc中100%的方法(不是函数)调用第一个参数都是self,同时,99%的情况下,调用方都不会在方法执行时把这个对象释放,所以相比于在每个方法中插入对self的引用计数管理:

- (void)start {

objc_retain(self);

// 其中的代码self一定不会被释放

objc_release(self);

}

优化了的性能还真是比较可观。

而且,ARC也用了挺多方法来避免开发者进行额外的引用计数控制,比如方法的命名约定,通过判断方法是否以如init,alloc,new,copy等关键字开头来决定其内存管理方式。

One more thing

在写test时发现,下面两种调用方法会导致不同结果:

- (void)viewDidLoad {

// 1

[_request start];// crash

// 2

[self.request start];// 正常

}

因为self.request是一次方法调用,返回的结果被objc_retainAutoreleasedReturnValue方法在局部进行了一次强引用,关于这个方法可以看之前写过的关于Autorelease的

转载地址:http://mvqko.baihongyu.com/

你可能感兴趣的文章
打印图片
查看>>
SHOW CREATE DATABASE Syntax
查看>>
rsync常见问题及解决办法
查看>>
MySQL日期 专题
查看>>
C#中禁止程序多开
查看>>
分布式缓存Redis使用以及原理
查看>>
Activity竟然有两个onCreate方法,可别用错了
查看>>
Linux经常使用命令(十六) - whereis
查看>>
core dump相关
查看>>
Linux五种IO模型
查看>>
Bootstrap技术: 模式对话框的使用
查看>>
小知识,用myeclipes找jar
查看>>
in-list expansion
查看>>
设计原则(四):接口隔离原则
查看>>
基于react的滑动图片验证码组件
查看>>
iOS快速清除全部的消息推送
查看>>
java单例模式深度解析
查看>>
【学习笔记】阿里云Centos7.4下配置Nginx
查看>>
VuePress手把手一小時快速踩坑
查看>>
dnsmasq安装使用和体验
查看>>