iOS为什么Block的属性声明要用copy

 我来答
世股年般1d
2017-09-14 · TA获得超过618个赞
知道小有建树答主
回答量:939
采纳率:0%
帮助的人:393万
展开全部
  1. Block的声明和线程安全

  2. Block属性的声明,首先需要用copy修饰符,因为只有copy后的Block才会在堆中,栈中的Block的生命周期是和栈绑定的,可以参考之前的文章(iOS: 非ARC下返回Block)。

  3. 另一个需要注意的问题是关于线程安全,在声明Block属性时需要确认“在调用Block时另一个线程有没有可能去修改Block?”这个问题,如果确定不会有这种情况发生的话,那么Block属性声明可以用nonatomic。如果不肯定的话(通常情况是这样的),那么你首先需要声明Block属性为atomic,也就是先保证变量的原子性(Objective-C并没有强制规定指针读写的原子性,C#有)。

  4. 比如这样一个Block类型:

  5. typedef void (^MyBlockType)(int);

属性声明:

@property (copy) MyBlockType myBlock;


这里ARC和非ARC声明都是一样的,当然注意在非ARC下要release Block。

但是,有了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];

推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式