从一道网易面试题浅谈OC线程安全

2017 年 8 月 26 日 CocoaChina Nemocdz

今天去网易面试,面试官出了一道面试题,下面代码会发生什么问题?


@property (nonatomic, strong) NSString *target;

//....


dispatch_queue_t queue = dispatch_queue_create("parallel", DISPATCH_QUEUE_CONCURRENT);

for (int i = 0; i < 1000000 ; i++) {

    dispatch_async(queue, ^{

        self.target = [NSString stringWithFormat:@"ksddkjalkjd%d",i];

    });

}


当时我把自定义的队列看成了串行队列,然后回答:“没错呀”。后来一运行崩溃了……


面试后,我就仔细回想,敲了Demo,看看崩溃原因是啥。


正好试试小伙伴给我介绍的调试野指针的方法,XCode7以上才有的Address Sanitizer


打开后发现是经典的EXC_BAD_ACCESS错误,以我浅薄的经验来看,这种一般是对一个已释放的内存的对象再次发送消息出现的。



再看看崩溃堆栈



噢,看来是对已释放的对象再次发送了release信息。


我又留意到,这个对象是Strong修饰的。


那么他的Setter方法在MRC上就相当于


- (void)setTarget:(NSString *)target {

    [target retain];//先保留新值

    [_target release];//再释放旧值

    _target = target;//再进行赋值

}


那么什么时候会导致过多调用release呢,因为这是个并行队列+异步。


那么假如队列A执行到步奏2,还没到步骤3时,队列B也执行到步骤2,那么这个对象就会被过度释放,导致向已释放内存对象发送消息而崩溃。


后来我想怎么可以修改这段代码变为不崩溃的呢?


1.使用串行队列


将set方法改成在串行队列中执行就行,这样即使异步,但所有block操作追加在队列最后依次执行。


2. 使用atomic


atomic关键字相当于在setter方法加锁,这样每次执行setter都是线程安全的,但这只是单独针对setter方法而言的狭义的线程安全。


3.使用weak关键字


weak的setter没有保留新值或者保留旧值的操作,所以不会引发重复释放。当然这个时候要看具体情况能否使用weak,可能值并不是所需要的值。


4.使用Tagged Pointer


Tagged Pointer是苹果在64位系统引入的内存技术。简单来说就是对于NSString(内存小于60位的字符串)或NSNumber(小于2^31),64位的指针有8个字节,完全可以直接用这个空间来直接表示值,这样的话其实会将NSString和NSNumber对象由一个指针转换成一个值类型,而值类型的setter和getter又是原子的,从而线程安全。


比如上述代码的字符串改短一些,就不会崩溃了。


从而我们可以总结到,线程安全有以下几种方法:


  • 单线程串行访问

  • 访问加锁

  • 使用不进行额外操作的关键字(weak)

  • 使用值类型


然而这只是保证了基本的线程安全(不崩溃),若是需要保证访问出符合预期的数据,则需要采用GCD的barrier或者自己在合适的时机加锁。


著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


登录查看更多
0

相关内容

面试是招聘、招生等的一个常见程序,指通过面谈来了解并评估应试者,来确定是否符合要求。
【2020新书】实战R语言4,323页pdf
专知会员服务
102+阅读 · 2020年7月1日
打怪升级!2020机器学习工程师技术路线图
专知会员服务
99+阅读 · 2020年6月3日
【实用书】Python爬虫Web抓取数据,第二版,306页pdf
专知会员服务
122+阅读 · 2020年5月10日
【2020新书】C++20 特性 第二版,A Problem-Solution Approach
专知会员服务
60+阅读 · 2020年4月26日
【干货书】流畅Python,766页pdf,中英文版
专知会员服务
226+阅读 · 2020年3月22日
【经典书】Python数据数据分析第二版,541页pdf
专知会员服务
195+阅读 · 2020年3月12日
在K8S上运行Kafka合适吗?会遇到哪些陷阱?
DBAplus社群
9+阅读 · 2019年9月4日
深度学习面试100题(第71-75题)
七月在线实验室
5+阅读 · 2018年8月2日
干货 | Python 爬虫的工具列表大全
机器学习算法与Python学习
10+阅读 · 2018年4月13日
机器学习面试题精讲(一)
七月在线实验室
4+阅读 · 2018年1月11日
python数据分析师面试题选
数据挖掘入门与实战
6+阅读 · 2017年11月21日
BAT题库 | 机器学习面试1000题系列(第196~200题)
七月在线实验室
17+阅读 · 2017年11月16日
机器学习/算法19家公司面试总结(内含薪资)
深度学习世界
12+阅读 · 2017年11月14日
[DLdigest-8] 每日一道算法
深度学习每日摘要
4+阅读 · 2017年11月2日
一个人的企业安全建设之路
FreeBuf
5+阅读 · 2017年7月7日
Do RNN and LSTM have Long Memory?
Arxiv
19+阅读 · 2020年6月10日
Arxiv
9+阅读 · 2018年3月23日
Arxiv
4+阅读 · 2018年3月14日
Arxiv
5+阅读 · 2015年9月14日
VIP会员
相关VIP内容
【2020新书】实战R语言4,323页pdf
专知会员服务
102+阅读 · 2020年7月1日
打怪升级!2020机器学习工程师技术路线图
专知会员服务
99+阅读 · 2020年6月3日
【实用书】Python爬虫Web抓取数据,第二版,306页pdf
专知会员服务
122+阅读 · 2020年5月10日
【2020新书】C++20 特性 第二版,A Problem-Solution Approach
专知会员服务
60+阅读 · 2020年4月26日
【干货书】流畅Python,766页pdf,中英文版
专知会员服务
226+阅读 · 2020年3月22日
【经典书】Python数据数据分析第二版,541页pdf
专知会员服务
195+阅读 · 2020年3月12日
相关资讯
在K8S上运行Kafka合适吗?会遇到哪些陷阱?
DBAplus社群
9+阅读 · 2019年9月4日
深度学习面试100题(第71-75题)
七月在线实验室
5+阅读 · 2018年8月2日
干货 | Python 爬虫的工具列表大全
机器学习算法与Python学习
10+阅读 · 2018年4月13日
机器学习面试题精讲(一)
七月在线实验室
4+阅读 · 2018年1月11日
python数据分析师面试题选
数据挖掘入门与实战
6+阅读 · 2017年11月21日
BAT题库 | 机器学习面试1000题系列(第196~200题)
七月在线实验室
17+阅读 · 2017年11月16日
机器学习/算法19家公司面试总结(内含薪资)
深度学习世界
12+阅读 · 2017年11月14日
[DLdigest-8] 每日一道算法
深度学习每日摘要
4+阅读 · 2017年11月2日
一个人的企业安全建设之路
FreeBuf
5+阅读 · 2017年7月7日
Top
微信扫码咨询专知VIP会员