美文网首页
使用Blocks来进行工作(三)

使用Blocks来进行工作(三)

作者: 进击的鸭子 | 来源:发表于2017-09-05 21:52 被阅读0次

一个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。

相关文章

网友评论

      本文标题:使用Blocks来进行工作(三)

      本文链接:https://www.haomeiwen.com/subject/udqpjxtx.html