一个Block永远是一个方法中的最后一个参数
最好的实践就是在一个方法中只使用一个block参数,如果这个方法同样需要其他非block的参数,那么这个block需要放在最后。
- (void)beginTaskWithName:(NSString *)name completion:(void(^)(void))callback;
当指定的block内嵌的时候,这种写法会使得该方法阅读起来更简单,像下面这样:
[self beginTaskWithName:@"MyTask" completion:^{
NSLog(@"this task is complete");
}];
使用类型定义来简化block的语法
如果你需要定义多个相同签名的block,你可能会为这个签名定义一个你独有的类型。
举个例子来说,你可以为一个简单的没有任何变量和返回值的block定义一个类型,像下面这样:
typedef void (^XYZSimpleBlock)(void);
然后你可以在这个方法参数中或者在创建block变量的时候,使用你的自定义类型:
XYZSimpleBlock anotherBlock = ^{
...
};
- (void)beginFetchWithCallbackBlock:(XYZSimpleBlock)callbackBlock {
...
callbackBlock();
}
自定义类型的定义实际上在处理blocks返回blocks或者获取其他blocks作为参数的时候非常有用。思考一下下面这个例子:
void (^(^complexBlock)(void(^)(void)))(void) = ^ (void (^aBlock)(void)) {
...
return ^ {
...
};
}
这个complexBlock变量指的是一个用其他block(aBlock)作为参数冰返回另一个block的block。重写代码来增加类型定义可读性:
XYZSimpleBlock (^betterBlock)(XYZSimpleBlock) = ^(XYZSimpleBlock aBlock) {
...
return ^{
...
};
}
Objects使用属性来持续跟踪Blocks
用定义属性来跟踪block和一个block变量的语法很相似:
@interface XYZObject : NSObject
@property (copy) void (^blockProperty)(void);
@end
注意:你要使用copy来修饰这个属性,因为我们需要复制block来跟踪它超出原始范围的捕捉状态。当使用自动引用计数的时候就不是一些你需要去担心的事情,因为它会自动发生,但是对于这个属性修饰来说最好的方式就是去展示行为结果。
一个block属性设置或者调用可以像任何其他block变量一样:
self.blockProperty = ^ {
...
};
self.blockProperty();
同样的可以用类型定义来对这个block属性进行修饰,像下面:
typedef void (^XYZSimpleBlock)(void);
@interface XYZObject : NSObject
@property (copy) XYZSimple blockProperty;
@end
当捕捉self的时候避免强引用循环
如果你需要去在block中捕捉self的时候,例如当定义一个callback的block,考虑内存管理的影响很重要。blocks对任何被捕捉的对象都持有一个强引用,包括self,这意味着它很容易结束强引用周期,例如,一个对象持有一个捕捉self的并且具有copy的属性的block:
@interface XYZSimpleBlockKeeper : NSObject
@property (copy) void (^block)(void);
@end
@implementation XYZBlockKeeper
- (void)configureBlock {
self.block = ^{
[self doSometing];
};
}
这个例子,编译器会发出警告,但是一个更复杂的例子可能涉及多个对象之间的强引用来创建循环,代码将会变得更加难诊断。为了避免这个问题,最好的办法就是捕捉一个self的弱引用,像下面这样:
- (void)configureBlock {
XYZBlockKeeper *__weak weakSelf = self;
self.block = ^ {
[weakSelf doSomething];
}
}
通过捕捉self的弱指针,block将不会持有与XYZBlockKeeper对象的强引用关系。如果这个对象在block调用之前被释放掉了,这个weakSelf指针将会很直接的就被重置成nil。
网友评论