您的位置:首页 > 移动开发 > IOS开发

ios 中的block应用

2016-01-23 15:52 381 查看
在这个大冬天里默默敲着键盘,勿喷.今天学习swift过程中,学习到闭包,发现闭包和oc的block中有很多的相同之处,又重新学习了一下并且学习了一些高级点的用法,内容如下:

1.block格式说明:(返回类型)(^块名称)(参数类型) = ^(参数列表) {代码实现};//如果没有参数,等号后面参数列表的()可以省略
例子:

void(^demoBlock)() = ^ {

NSLog(@"demo Block");

};

int(^sumBlock)(int, int) = ^(int x, int y) {

return x + y;

};

2.block中使用的变量将以复制的形式保留,在block中保留block的复制的变量,默认情况下,Block外部的变量,在Block中是只读的!当使用__block关键字后,同样可以在Block中修改外部变量的数值.

3.可以使用typedef定义一个Block的类型,便于在后续直接使用,如:
typedef double(^MyBlock)(double, double);

MyBlock area = ^(double x, double y) {

return x * y;

};

MyBlock sum = ^(double a, double b) {

return a + b;

};

NSLog(@"%.2f", area(10.0, 20.0));

NSLog(@"%.2f", sum(10.0, 20.0));

4.尽管,typedef可以简化Block的定义,但在实际开发中并不会频繁使用typedef关键字,这是因为Block具有非常强的灵活性,尤其在以参数传递时,使用Block的目的就是为了立即使用
官方的数组遍历方法声明如下:
- (void)enumerateObjectsUsingBlock:(void (^)(id obj, NSUInteger idx, BOOL *stop))block;

而如果使用typedef,则需要:
(1)typedef void(^EnumerateBlock)(id obj, NSUInteger idx, BOOL *stop);
(2)- (void)enumerateObjectsUsingBlock:(EnumerateBlock)block;
而最终的结果却是,除了定义类型之外,EnumerateBlock并没有其他用处

5.Block可以被当做参数直接传递

NSArray *array = @[@"张三", @"李四", @"王五", @"赵六"];

[array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {

NSLog(@"第 %d 项内容是 %@", (int)idx, obj);

if ([@"王五" isEqualToString:obj]) {

*stop = YES;

}

}];

或者:

MyBlock sumBlock = ^(double x, double y) {
return x * y;

};

- (void)add:(int)number withNumber:(int)withNumber sumBlock:(MyBlock) block
{

NSLog(@"执行add:(int)number withNumber:(int)withNumber sumBlock:(void (^)(void))block");
NSLog(@"hahaha--4和2相乘等于:%f",block(number,withNumber));

}

[self add:4 withNumber:2 sumBlock:sumBlock];

6.既然Block是一种数据类型,那么可以将Block当做比较特殊的对象,添加到数组

#pragma mark 定义并添加到数组

@property (nonatomic, strong) NSMutableArray *myBlocks;

int(^sum)(int, int) = ^(int x, int y) {

return [self sum:x y:y];

};

[self.myBlocks addObject:sum];

int(^area)(int, int) = ^(int x, int y) {

return [self area:x y:y];

};

[self.myBlocks addObject:area];

#pragma mark 调用保存在数组中的Block

int(^func)(int, int) = self.myBlocks[index];

return func(x, y);

7.解除循环引用

局部变量默认都是强引用的,离开其所在的作用域之后就会被释放

使用__weak关键字,可以将局部变量声明为弱引用
__weak DemoObj *weakSelf = self;

n在Block中引用weakSelf,则Block不会再对self做强引用
int(^sum)(int, int) = ^(int x, int y) {

return [weakSelf sum:x y:y];

};

8.今天重点,参考AFN的框架,block根据函数判断正确与否调用success或者failure函数参数,并传入参数.如下:

//由于主要还是用来判断success和failure函数,所以看着不舒服的同学可以自行脑补,将success换成add,将failure换成multiplied

- (void)viewDidLoad {
[super viewDidLoad];
self.isbool = true;//当isbool等于true的时候就相加,否则就相乘

[self setCompletionBlockWithSuccess:^(int x,int y) {
NSLog(@"相加等于:%d",x+y);

NSLog(@"success");

} failure:^(int x,int y) {

NSLog(@"相乘等于:%d",x*y);
NSLog(@"failure");

}];

}

- (void)setCompletionBlock:(void (^)(void))block {

block();
NSLog(@"setCompletionBlock number2:%d number3:%d",self.number2,self.number3);
}

- (void)setCompletionBlockWithSuccess:(void (^)(int x,int y))success
failure:(void (^)(int x,int y))failure
{

self.completionBlock = ^{

self.number2 = 10;
self.number3 = 20;

if (self.isbool == true) {//判断传入哪个参数,执行success还是failure
NSLog(@"开始相加");
success(self.number2,self.number3);//参数传入setCompletionBlockWithSuccess
}else{
NSLog(@"开始相乘");
failure(self.number2,self.number3);//参数传入setCompletionBlockWithSuccess
}
};

}

9.block遍历

enumerateObjectsUsingBlock:^(id obj, NSUInteger idx,BOOL *stop){ }
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: