简介:
block是代码块,其本质和变量类似。不同的是代码块存储的数据是一个函数体。block使用最多的是存储代码块和回调。
一、注意事项:
1. 在使用block前需要对block指针做判空处理。block的声明类判断,非调用类判断。
2. block变量定义时为什么用copy?block是放在哪里的?
只有copy后才会放到堆上。 默认情况下,block是存档在栈中,可能被随时回收,需要copy操作。这也就是我们在定义block的时候用得时copy。而不是weak等等。
3. block内部只能访问局部变量,不能修改。避免循环引用,内部使用时要用weak处理:
__weak typeof(self) weakSelf = self;
严格意义上在block内部,self存在被释放的可能,严格的写法是在内部还原回strong:
__strong typeof(weakSelf) strongSelf = weakSelf;
例如:
__weak typeof(self) weakSelf = self;
[self doSomeBlock:^{
__strong typeof(weakSelf) strongSelf = weakSelf;
if(strongSelf) {
...
}
}];
二、定义格式:
返回值类型(^block变量名)(形参列表) = ^(形参列表) {
};
三、调用方式:
block变量名(实参);
//可以作为属性调用,也可以以方法形式调用,顺便dosomeThing
- (void)handlerButtonAction:(GetValueBlock)block
{
self.getValueBlock= block;
//doSomeThing
}
为空判断:
if (self.getValueBlock) {
self.getValueBlock(self.textField.text);
}
四、举个例子:
问题,A视图 push 到B视图,然后从B回到A,这个时候做了一些操作想让A产生反应。你可能会想到通知,协议。前者就不说了,不推荐使用。协议也可以,但是协议的功能性更强大。一般简单的block就可以完成。
那么我们就需要在B视图定义好block,并且在B视图返回的时候把参数传递给A。看代码:
首先,两种创建方式:
typedef void (^ReturnTextBlock)(NSString *showText);
@property (copy, nonatomic) ReturnTextBlock messageBlock;
@property (copy, nonatomic) NSString *(^logBlock)(NSString *first,NSString *second);
其实上面的代码也很明显,已经给出了block的不同形式和写法了。大致上三种:
无参无返回值
有参无返回值
有参有返回值
回到问题,接下来是在B视图返回的时候做一些操作。demo里面是写了个textfield,然后点击button触发back把textfield里的内容回调给A
对于有参无返回值是我用的最多的,也是比较简单的一种。 无参无返回值的也请参考这个。
- (IBAction)blockBack:(id)sender {
self.messageBlock(self.numberTextField.text);
[self.navigationController popViewControllerAnimated:YES];
}
调用方法:
BVC.messageBlock = ^(NSString *first){
self.pushBtn.titleLabel.text = first;
};
这个其实比较简单,然后说下有参数有返回值的,也是我遇到的问题。比较痛苦的是翻遍了百度找不到好的答案。
先写BVC中block的实现方法:
- (IBAction)blockBackTwo:(id)sender {
NSString *back = self.logBlock(self.numberTextField.text,self.secondTextField.text);
//拿到A界面运算后的返回值给B
NSLog(@"back:%@",back);
[self.navigationController popViewControllerAnimated:YES];
}
AVC中的调用方法:
vc.logBlock = ^NSString *(NSString *first, NSString *second) {
NSString *addStr = [first stringByAppendingString:second];
self.pushBtn.titleLabel.text = addStr;
return addStr;
};
解释:
handBack是封装block的方法,效果是一样的。可以忽视。
问题主要在于对有返回值的block的理解。一开始不知道怎么写这个返回值。思考了半天,也请教了很多人,然后发现大家都是似懂非懂,给我的解决办法也都不管用。最后一边失败一边思考,琢磨出来这种写法。其实失败也是自己对block的理解不到位。
首先这里block的返回值是用于B(block实现界面)的,也就是说我B界面把参数给了A,A进行操作以后得到一个值,这个值给我,我再做处理。相当于两次block其实,双重回调。理解了这个就好做多了。
五、使用__weak修饰和使用__Block修饰
__weak本身是可以避免循环引用的问题的,但是其会导致外部对象释放了之后,block 内部也访问不到这个对象的问题,我们可以通过在 block 内部声明一个__strong的变量来指向 weakObj,使外部对象既能在 block 内部保持住,又能避免循环引用的问题。
__block本身无法避免循环引用的问题,但是我们可以通过在 block
内部手动把 blockObj 赋值为 nil 的方式来避免循环引用的问题。另外一点就是__block修饰的变量在 block
内外都是唯一的,要注意这个特性可能带来的隐患。__block其实提升了变量的作用域,有点类似于Static。