22

编辑2:

不,建议的答案是关于异步调用。我想要并需要同步调用,就像在正常的标准递归调用中一样。

编辑:

尽管

__unsafe_unretained void (^unsafe_apply)(UIView *, NSInteger) ;

编译时没有警告或错误,它在运行时失败,并将 NULL 存储到 unsafe_apply 中。

然而这:

- (void) applyToView: (UIView *) view {

    UIColor * (^colorForIndex)(NSInteger) = ^(NSInteger index) {
        return [UIColor colorWithHue: ((CGFloat) index / 255.0f)
                          saturation: 0.5f
                          brightness: 0.5f
                               alpha: 1.0f] ;
    } ;

    void (^applyColors) (UIView *, NSInteger index) = ^(UIView * view, NSInteger index) {
        view.backgroundColor = colorForIndex(index) ;
    } ;

    void (^__block recurse_apply)(UIView *, NSInteger) ;

    void (^apply)(UIView *, NSInteger) = ^(UIView * view, NSInteger level) {
        applyColors(view, level) ;
        [view.subviews enumerateObjectsUsingBlock:^(UIView * subview, NSUInteger idx, BOOL *stop) {
            recurse_apply(subview, 1+level) ;
        }] ;
    } ;

    recurse_apply = apply ;

    apply(view, 0) ;
}

编译没有警告,但更重要的是,实际运行。

但这太丑了


考虑(着色视图层次结构,用于公开目的......)

- (void) applyToView: (UIView *) view {

    UIColor * (^colorForIndex)(NSInteger) = ^(NSInteger index) {
        return [UIColor colorWithHue: ((CGFloat) (index * 10.0f) / 255.0f)
                          saturation: 0.5f
                          brightness: 0.5f
                               alpha: 1.0f] ;
    } ;

    void (^applyColors) (UIView *, NSInteger index) = ^(UIView * view, NSInteger index) {
        view.backgroundColor = colorForIndex(index) ;
    } ;

    void (^apply)(UIView *, NSInteger) = ^(UIView * view, NSInteger level) {
        applyColors(view, level) ;
        [view.subviews enumerateObjectsUsingBlock:^(UIView * subview, NSUInteger idx, BOOL *stop) {
            apply(subview, 1+level) ;
        }] ;
    } ;

    apply(view, 0) ;
}

我收到这个警告:

/Users/verec/Projects/solotouch/SoloTouch/BubbleMenu.m:551:42:Block pointer variable 'apply' is uninitialized when captured by block

如果我应用建议的修复:Maybe you meant to use __block 'apply'

void (^__block apply)(UIView *, NSInteger) = ^(UIView * view, NSInteger level) {

然后我得到:/Users/verec/Projects/solotouch/SoloTouch/BubbleMenu.m:554:13:Capturing 'apply' strongly in this block is likely to lead to a retain cycle

我尝试了各种方法来篡改代码并摆脱那些警告

__weak typeof (apply) wapply = apply ;
if (wapply) {
    __strong typeof (wapply) sappy = wapply ;
    wapply(subview, 1+level) ;
}

但事情只会变得更糟,变成错误。

我最终得到了这个:

__unsafe_unretained void (^unsafe_apply)(UIView *, NSInteger) ;

void (^apply)(UIView *, NSInteger) = ^(UIView * view, NSInteger level) {
    applyColors(view, level) ;
    [view.subviews enumerateObjectsUsingBlock:^(UIView * subview, NSUInteger idx, BOOL *stop) {
        unsafe_apply(subview, 1+level) ;
    }] ;
} ;

unsafe_apply = apply ;

apply(view, 0) ;

任何人都有更好的解决方案,我可以在块内做所有事情,而不是像我在这里必须做的那样可怕地修补它?

注意那些SO 问题是关于捕获self的,那些SO 问题没有任何令人满意的答案。

4

3 回答 3

64

您需要捕获一个__block变量,因为块__block在创建时按值捕获非变量,并且分配发生创建块之后。

在 ARC 中,__block对象指针类型的变量(通常所有变量都是隐式__strong的)由块保留。因此,如果块捕获__block指向自身的变量,它将创建一个保留循环。解决方案是让它捕获一个弱引用。在支持的操作系统版本中__weak__weak应使用 , 而不是__unsafe_unretained.

但是,如果对块的唯一引用是__weak变量,则不会有对块的强引用,这意味着它可以被释放。为了使用块,它必须有一个强引用来保持它。

因此,您需要两个变量,一弱一强。在 ARC 中执行此操作的正确方法是:

__block __weak void (^weak_apply)(UIView *, NSInteger) ;
void (^apply)(UIView *, NSInteger) ;
weak_apply = apply = ^(UIView * view, NSInteger level) {
    applyColors(view, level) ;
    [view.subviews enumerateObjectsUsingBlock:^(UIView * subview, NSUInteger idx, BOOL *stop) {
        weak_apply(subview, 1+level) ;
    }] ;
} ;

apply(view, 0) ;
于 2013-11-11T11:38:13.183 回答
1

为了避免 ARC 警告并建立在 @newacct 的答案上,我发现在保留块内设置弱块是有效的:

//a block definition
typedef void (^CompletionType)(NSDictionary * __nullable response, NSError * __nullable error);


//a block calling itself without the ARC retain warning
__block CompletionType completionBlock = nil;
__block __weak CompletionType weakCompletionBlock = nil;
completionBlock = ^(NSDictionary *response, NSError *error) {
    weakCompletionBlock = completionBlock;

        weakCompletionBlock();

    });
};
于 2016-07-06T10:24:36.867 回答
0

答案是不。

我们似乎不能比使用 __block 限定符做得更好。

__block void(^strawberryFields)();
strawberryFields = ^{ strawberryFields(); };
strawberryFields();

感谢Bill Bumgarner关于区块的文章。


编辑:

__block __weak void(^strawberryFields)();

似乎是ARC下的首选方式。

于 2013-11-10T02:23:22.283 回答