iOS: ARC和非ARC下使用Block与循环引用问题
2016-01-21 19:45
363 查看
1. Block的声明和线程安全
Block属性的声明,首先需要用copy修饰符,因为只有
copy后的Block才会在堆中,栈中的Block的生命周期是和栈绑定的,可以参考之前的文章(iOS: 非ARC下返回Block)。
另一个需要注意的问题是关于线程安全,在声明Block属性时需要确认“在调用Block时另一个线程有没有可能去修改Block?”这个问题,如果确定不会有这种情况发生的话,那么Block属性声明可以用
nonatomic。如果不肯定的话(通常情况是这样的),那么你首先需要声明Block属性为
atomic,也就是先保证变量的原子性(Objective-C并没有强制规定指针读写的原子性,C#有)。
比如这样一个Block类型:
typedef void (^MyBlockType)(int);
属性声明:
@property (copy) MyBlockType myBlock;
这里ARC和非ARC声明都是一样的,当然注意在非ARC下要
releaseBlock。
但是,有了
atomic来保证基本的原子性还是没有达到线程安全的,接着在调用时需要把Block先赋值给本地变量,以防止Block突然改变。因为如果不这样的话,即便是先判断了Block属性不为空,在调用之前,一旦另一个线程把Block属性设空了,程序就会crash,如下代码:
if (self.myBlock) { //此时,走到这里,self.myBlock可能被另一个线程改为空,造成crash //注意:atomic只会确保myBlock的原子性,这种操作本身还是非线程安全的 self.myBlock(123); }
所以正确的代码是(ARC):
MyBlockType block = self.myBlock; //block现在是本地不可变的 if (block) { block(123); }
在非ARC下则需要手动
retain一下,否则如果属性被置空,本地变量就成了野指针了,如下代码:
//非ARC MyBlockType block = [self.myBlock retain]; if (block) { block(123); } [block release];
返回目录
2. 循环引用问题
循环引用是另一个使用Block时常见的问题。在ARC下,由于
__block抓取的变量一样会被Block
retain,所以必须用弱引用才可以解决循环引用问题,iOS 5之后可以直接使用
__weak,之前则只能使用
__unsafe_unretained了,
__unsafe_unretained缺点是指针释放后自己不会置空。示例代码:
//iOS 5之前可以用__unsafe_unretained //__unsafe_unretained typeof(self) weakSelf = self; __weak typeof(self) weakSelf = self; self.myBlock = ^(int paramInt) { //使用weakSelf访问self成员 [weakSelf anotherFunc]; };
在非ARC下,显然无法使用弱引用,这里就可以直接使用
__block来修饰变量,它不会被Block所
retain的,参考代码:
//非ARC __block typeof(self) weakSelf = self; self.myBlock = ^(int paramInt) { //使用weakSelf访问self成员 [weakSelf anotherFunc]; };
相关文章推荐
- StoryBoard设置Cell中的Label自适应高度
- iOS中的图形
- iOS-NSDate
- iOS AFNetworking 图片上传(修改用户头像功能)
- iOS实现图像的反色,怀旧,色彩直方图效果
- iOS开发之详解剪贴板
- iOS 忽略SIGPIPE的问题
- iOS 导航栏按钮封装
- Xcode如何删除已安装的插件
- iOS - block的简单使用
- ios中代码打开app store
- iOS 单击手势和双击手势冲突问题
- iOS安全笔记
- 线程与进程的区别和联系
- IOS常用正则表达式
- iOS开发--代码块 (得到状态栏,导航栏高度)
- iOS开发之Xcode6之后不再自动创建Pch预编译文件
- iOS block简单使用
- IOS中定时器NSTimer的开启与关闭
- IOS8之后ActionSheet和ActionView合体,简单更好用