Block简介:
block
其实就是一个代码块,把你想要执行的代码封装在这个代码块里,等到需要的时候再去调用。那block是OC对象吗?答案是肯定的
修饰符
简单来说,block就像一个函数指针,指向我们要使用的函数。
就和函数调用一样的,不管你在哪里写了这个block,只要你把它放在了内存中(通过调用存在这个block的方法或者是函数),不管放在栈中还是在堆中,还是在静态区。只要他没有被销毁,你都可以通过你声明的block调用他。
说到在类中声明一个block为什么要用copy修饰的话,那就要先说block的三种类型。
1.NSConcreteGlobalBlock 全局的静态block,不会访问外部的变量。就是说如果你的block没有调用其他的外部变量,那你的block类型就是这种。例如:你仅仅在你的block里面写一个NSLog(“hello world”);
2.NSConcreteStackBlock 保存在栈中的 block,当函数返回时会被销毁。这个block就是你声明的时候不用copy修饰,并且你的block访问了外部变量。
3.NSConcreteMallocBlock 保存在堆中的block,当引用计数为 0 时会被销毁。好了,这个就是今天的主角 ,用copy修饰的block。
我们知道,函数的声明周期是随着函数调用的结束就终止了。我们的block是写在函数中的。
如果是全局静态block的话,他直到程序结束的时候,才会被被释放。但是我们实际操作中基本上不会使用到不访问外部变量的block。【但是在测试三种区别的时候,因为没有很好的理解这种block,(用没有copy修饰和没有访问外部变量的block)试了好多次,以为是放在静态区里面的block没有随函数结束被释放。这是个小坑】
如果是保存在栈中的block,他会随着函数调用结束被销毁。从而导致我们在执行一个包含block的函数之后,就无法再访问这个block。因为(函数结束,函数栈就销毁了,存在函数里面的block也就没有了),我们再使用block时,就会产生空指针异常。
如果是堆中的block,也就是copy修饰的block。他的生命周期就是随着对象的销毁而结束的。只要对象不销毁,我们就可以调用的到在堆中的block。
这就是为什么我们要用copy来修饰block。因为不用copy修饰的访问外部变量的block,只在他所在的函数被调用的那一瞬间可以使用。之后就消失了。
使用retain也可以,但是block的retain行为默认是用copy的行为实现的,
因为block变量默认是声明为栈变量的,为了能够在block的声明域外使用,所以要把block拷贝(copy)到堆,所以说为了block属性声明和实际的操作一致,最好声明为copy。