原文: http://wzxjiang.com/2016/05/12/weakify-and-strongify/?utm_source=tuicool&utm_medium=referral
使用rac的都知道@weakify和@strongify是在它擴(kuò)庫libextobjc中的一個(gè)宏,通過他可以實(shí)現(xiàn)__weak和__strong的效果亏狰,那它使怎么實(shí)現(xiàn)的呢役纹。
@autoreleasepool {} __attribute__((objc_ownership(weak))) __typeof__(self) self_weak_ = (self);;
__attribute__((objc_ownership(strong))) __typeof__(self) self = self_weak_;
@autoreleasepool {} 實(shí)現(xiàn)了添加@的效果,attribute((objc_ownership(weak)))就是weak的實(shí)現(xiàn)暇唾,_typeof取類型促脉,相當(dāng)與創(chuàng)建了弱引用self的self_weak的局部變量。attribute((objc_ownership(strong))) 相當(dāng)于_strong策州,創(chuàng)建了對self_weak的強(qiáng)引用瘸味,變量名為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),有多個(gè)參數(shù)下硕,第一個(gè)參數(shù)是 CONTEXT typeof(VAR) metamacro_concat(VAR, _weak) = (VAR);第二個(gè)沒有丁逝,第三個(gè)是weak汁胆,最后是多參數(shù)的意思,代表self,缺省號代表一個(gè)可以變化的參數(shù)表梭姓。使用保留名 __VA_ARGS 把參數(shù)傳遞給宏。當(dāng)宏的調(diào)用展開時(shí)嫩码,實(shí)際的參數(shù)就傳遞給metamacro_foreach_cxt了
第一個(gè)參數(shù)不展開變成如下樣子:
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的意思是連接第一個(gè)和第二個(gè)參數(shù)誉尖,因?yàn)閰?shù)數(shù)量是(參數(shù)的計(jì)算后面有解釋)1,所以變成如下樣子:
metamacro_foreach_cxt1(ext_weakify_,,__weak,self)
metamacro_foreach_cxt1分解后這樣子:
metamacro_foreach_cxt1(ext_weakify_,,__weak,self)
由于MACRO(0, CONTEXT, _0)繼續(xù)分解:
第一個(gè)參數(shù)是個(gè)宏把__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計(jì)算參數(shù)個(gè)數(shù)琢感,文檔說返回提供給該宏的參數(shù)個(gè)數(shù)(超過20個(gè)),至少一個(gè)必須提供探熔,展開看下:
#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__)
因?yàn)閙etamacro_concat展開是A ## B連接的意思驹针,metamacro_at連接(拼接)參數(shù)N于是變成如下樣子:
metamacro_at20(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19,...)
截?cái)嗔饲懊?0個(gè)參數(shù)因?yàn)?..此時(shí)代表1
展開是這個(gè):metamacro_head(1),后面metamacro_head_(FIRST, ...) FIRST诀艰,因?yàn)榉祷氐谝粋€(gè)柬甥,所以參數(shù)參數(shù)個(gè)數(shù)是1