美文网首页iOS 基础
Reactive Cocoa中@weakify和@strongi

Reactive Cocoa中@weakify和@strongi

作者: 开心小锣鼓 | 来源:发表于2016-06-14 18:07 被阅读283次

    原文: http://wzxjiang.com/2016/05/12/weakify-and-strongify/?utm_source=tuicool&utm_medium=referral

    使用rac的都知道@weakify和@strongify是在它扩库libextobjc中的一个宏,通过他可以实现__weak和__strong的效果,那它使怎么实现的呢。

    @autoreleasepool {} __attribute__((objc_ownership(weak))) __typeof__(self) self_weak_ = (self);;

    __attribute__((objc_ownership(strong))) __typeof__(self) self = self_weak_;

    @autoreleasepool {} 实现了添加@的效果,attribute((objc_ownership(weak)))就是weak的实现,_typeof取类型,相当与创建了弱引用self的self_weak的局部变量。attribute((objc_ownership(strong))) 相当于_strong,创建了对self_weak的强引用,变量名为self。

    宏展开以@weakify为例:

    第一层:

    #define weakify(...) \

    ext_keywordify \

    metamacro_foreach_cxt(ext_weakify_,, __weak, __VA_ARGS__)

    分析:1、ext_keywordify autoreleasepool {}可以添加@

    2、metamacro_foreach_cxt(ext_weakify_,, weak, _VA_ARGS),有多个参数,第一个参数是 CONTEXT typeof(VAR) metamacro_concat(VAR, _weak) = (VAR);第二个没有,第三个是weak,最后是多参数的意思,代表self,缺省号代表一个可以变化的参数表。使用保留名 __VA_ARGS 把参数传递给宏。当宏的调用展开时,实际的参数就传递给metamacro_foreach_cxt了

    第一个参数不展开变成如下样子:

    metamacro_foreach_cxt(ext_weakify_,,__weak,self)

    第二层:

    分解metamacro_foreach_cxt

    #define metamacro_foreach_cxt(MACRO, SEP, CONTEXT, ...) \

    metamacro_concat(metamacro_foreach_cxt, metamacro_argcount(__VA_ARGS__))(MACRO, SEP, CONTEXT, __VA_ARGS__)

    metamacro_concat的意思是连接第一个和第二个参数,因为参数数量是(参数的计算后面有解释)1,所以变成如下样子:

    metamacro_foreach_cxt1(ext_weakify_,,__weak,self)

    metamacro_foreach_cxt1分解后这样子:

    metamacro_foreach_cxt1(ext_weakify_,,__weak,self)

    由于MACRO(0, CONTEXT, _0)继续分解:

    第一个参数是个宏把__weak,self放入MACRO(0, CONTEXT, _0)

    如下:

    weak _typeof(self) self_weak = (self);

    strong的原理与此差不多,这里不再赘述。

    OK!!完成

    metamacro_concat(metamacro_foreach_cxt, metamacro_argcount(__VA_ARGS__))(MACRO, SEP, CONTEXT, __VA_ARGS__)

    metamacro_argcount计算参数个数,文档说返回提供给该宏的参数个数(超过20个),至少一个必须提供,展开看下:

    #define metamacro_argcount(...) \

    metamacro_at(20, self, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

    metamacro_at展开如下:

    #define metamacro_at(20, self) \

    metamacro_concat(metamacro_at, N)(__VA_ARGS__)

    因为metamacro_concat展开是A ## B连接的意思,metamacro_at连接(拼接)参数N于是变成如下样子:

    metamacro_at20(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19,...)

    截断了前面20个参数因为...此时代表1

    展开是这个:metamacro_head(1),后面metamacro_head_(FIRST, ...) FIRST,因为返回第一个,所以参数参数个数是1

    相关文章

      网友评论

        本文标题:Reactive Cocoa中@weakify和@strongi

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