美文网首页iOS Developer - TipsiOS程序员
请谨慎使用 @weakify 和 @strongify

请谨慎使用 @weakify 和 @strongify

作者: 酷酷的哀殿 | 来源:发表于2016-09-05 13:30 被阅读4838次

    前言

    相信大部分见过 @weakify 和 @strongify 的开发者都会喜欢上这两个宏。但是很多人只知道它的强大威力,却没有意识到在特定环境下的危险性。

    本文将通过代码测试的方式告诉读者,如何正确地使用这两个的宏。

    @weakify 和 @strongify

    本文意在说明其危险性,所以不会全面的讲解这两个宏。
    如果您对其该兴趣,请参考其它作者的文章或者自行查看源码。

    这两个宏的定义如下:

    EXTScope.h#L45-L47

    #define weakify(...) \
        rac_keywordify \
        metamacro_foreach_cxt(rac_weakify_,, __weak, __VA_ARGS__)
    

    EXTScope.h#L83-L88

    #define strongify(...) \
        rac_keywordify \
        _Pragma("clang diagnostic push") \
        _Pragma("clang diagnostic ignored \"-Wshadow\"") \
        metamacro_foreach(rac_strongify_,, __VA_ARGS__) \
        _Pragma("clang diagnostic pop")
    

    其中 rac_keywordify 的定义如下:
    EXTScope.h#L114-L118

    #if DEBUG
    #define rac_keywordify autoreleasepool {}
    #else
    #define rac_keywordify try {} @catch (...) {}
    #endif
    

    测试

    下面是官方提供了一个示例代码。
    示例代码中定义了一个 block,该 block 用于判断入参 obj 是否和 foofar 其中的任何一个对象相等并返回 YESNO

        id foo = [[NSObject alloc] init];
        id bar = [[NSObject alloc] init];
    
        @weakify(foo, bar);
    
        // this block will not keep 'foo' or 'bar' alive
        BOOL (^matchesFooOrBar)(id) = ^ BOOL (id obj){
            // but now, upon entry, 'foo' and 'bar' will stay alive until the block has
            // finished executing
            @strongify(foo, bar);
    
            return [foo isEqual:obj] || [bar isEqual:obj];
        };
    

    测试代码一

    为了方便测试,这里重写了 rac_keywordify 的定义。

        {
        #undef rac_keywordify
    
        #define rac_keywordify autoreleasepool { }
    
            id foo = [[NSObject alloc] init];
            id bar = [[NSObject alloc] init];
    
            @weakify(foo, bar);
    
            BOOL (^matchesFooOrBar)(id) = ^ BOOL (id obj){
                @strongify(foo, bar);
                NSLog(@"%@,%@", foo, bar);
            };
        }
    

    相信眼尖的读者一眼就能看出与上面代码的不同。
    block缺少返回值

    下面是 Xcode 的截图。Xcode 产生一个 Control reaches end of non-void block 的❗️错误提示。

    错误提示.png

    测试代码二

    为了方便测试,这里重写了 rac_keywordify 的定义。

        {
            #undef rac_keywordify
    
            #define rac_keywordify try { } @catch(...) {}
    
            id foo = [[NSObject alloc] init];
            id bar = [[NSObject alloc] init];
    
            @weakify(foo, bar);
    
            BOOL (^matchesFooOrBar)(id) = ^ BOOL (id obj){
                @strongify(foo, bar);
                NSLog(@"%@,%@", foo, bar);
            };
        }
    

    这份代码除了将 #define rac_keywordify autoreleasepool { } 修改为 #define rac_keywordify try { } @catch(...) {}以外,与上面的代码并没有不同。
    理想的情况当然时,Xcode 依然有❗️错误提示。但是,现实往往是残酷的,Xcode 只提供了一个未使用变量的⚠️。

    无错误提示.png

    由上图可知,Xcode 丢失了错误提示的能力

    问题分析

    在 Release 模式下,rac_keywordify 被定义为 #define rac_keywordify try { } @catch(...) {},经预处理器处理后,会转换为下面的代码

            id foo = [[NSObject alloc] init];
            id bar = [[NSObject alloc] init];
    
            @try { } @catch(...) {} __attribute__((objc_ownership(weak))) __typeof__(foo) foo_weak_ = (foo); __attribute__((objc_ownership(weak))) __typeof__(bar) bar_weak_ = (bar);;
    
            BOOL (^matchesFooOrBar)(id) = ^ BOOL (id obj){
                @try { } @catch(...) {}
    # 99 "/Users/L/Documents/workspace/.../AppDelegate.m"
    #pragma clang diagnostic push
    # 99 "/Users/L/Documents/workspace/.../AppDelegate.m"
    #pragma clang diagnostic ignored "-Wshadow"
    # 99 "/Users/L/Documents/workspace/.../AppDelegate.m"
     __attribute__((objc_ownership(strong))) __typeof__(foo) foo = foo_weak_; __attribute__((objc_ownership(strong))) __typeof__(bar) bar = bar_weak_;
    # 99 "/Users/L/Documents/workspace/.../AppDelegate.m"
    #pragma clang diagnostic pop
    # 99 "/Users/L/Documents/workspace/.../AppDelegate.m"
    ;
                NSLog(@"%@,%@", foo, bar);
            };
    

    @try { } @catch(...) {}被添加到了等式的前面。
    在这种情况下,Xcode 本身的错误提示能力能被抑制了,就如同源码的注释中提到的那样。

    // Details about the choice of backing keyword:
    //
    // The use of @try/@catch/@finally can cause the compiler to suppress
    // return-type warnings.
    // The use of @autoreleasepool {} is not optimized away by the compiler,
    // resulting in superfluous creation of autorelease pools.
    //
    // Since neither option is perfect, and with no other alternatives, the
    // compromise is to use @autorelease in DEBUG builds to maintain compiler
    // analysis, and to use @try/@catch otherwise to avoid insertion of unnecessary
    // autorelease pools.
    #if DEBUG
    #define rac_keywordify autoreleasepool {}
    #else
    #define rac_keywordify try {} @catch (...) {}
    #endif
    

    很多人都研究过这部分代码,但是大部分的人都得出类似于这样的结论。

    这段宏定义中的代码开头都少了个@,使得weakify、strongify前面必须加上@,当然也只有这作用。 然后这里为什么要判断DEBUG呢?我也不知道,我觉得这里没必要这样判断。

    判断DEBUG的作用在于,正常的开发模式都是在DEBUG模式下面进行的。这样可以保留 Xcode 提示错误的能力

    结论

    请读者回想一下,你是否可以快速的判断出自己是否在 DEBUG模式下开发?如果回答是NO,请谨慎使用 @weakify@strongify

    修改开发模式

    点击项目名称,在弹出框中,点击 Edit Scheme...

    Paste_Image.png

    在模态视图中,点击 Build Configuration 单选框

    Paste_Image.png

    相关文章

      网友评论

      • 苦工:用这个宏@weakify(self);会报错 Unexpected '@' in program 咋回事啊
      • Lin__Chuan:说了这么多,最基础的,这两个宏是干嘛的,这个没解释
        Lin__Chuan:@酷酷的哀殿 OK
        酷酷的哀殿:@MarisLin 自己百度吧,这篇文章的目的不是介绍这个的
      • 牧马:不知道想表达什么,标题取得像 UC 震惊部,内容却不知所云
        酷酷的哀殿:@牧马 我的标题是“谨慎”不是“禁止”,release 也可以断点、调试,这个没有问题
        牧马:@酷酷的哀殿 你见过在开发模式用 Release 的?那你怎么打断点,怎么调试?不要夸大其词 危言耸听
        酷酷的哀殿:静下心来,再看
      • sun猩猩:可以借鉴一下YYKit的对于这2个宏的写法
      • 子达如何:怎么没看懂说什么呢……
        酷酷的哀殿:@子达如何 在RELEASE模式下是try
        子达如何:@酷酷的哀殿 那是release版本才是try呀
        酷酷的哀殿: @子达如何 在使用@try 语法后,会导致编译器在没有返回值时也无法报错
      • 一缕殇流化隐半边冰霜::+1::+1:涨知识了!设计这个宏的时候还考虑到了调试的便捷:+1::+1:

      本文标题:请谨慎使用 @weakify 和 @strongify

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