Reactive Cocoa Tutorial [1] = "神奇的Macros";


  Reactive Cocoa Tutorial 系列,轉載請注明該文源地址 -- by sunny


  先說說RAC中必須要知道的宏:

  · RAC(TARGET, [KEYPATH, [NIL_VALUE]])

RAC(self.outputLabel, text) = self.inputTextField.rac_textSignal;

RAC(self.outputLabel, text, @"收到nil時就顯示我") = self.inputTextField.rac_textSignal;

  這個宏是最常用的,RAC()總是出現在等號左邊,等號右邊是一個RACSignal,表示的意義是將一個對象的一個屬性和一個signal綁定,signal每產生一個value(id類型),都會自動執行:

[TARGET setValue:value ?: NIL_VALUE forKeyPath:KEYPATH];

  數字值會升級為NSNumber *,當setValue:forKeyPath時會自動降級成基本類型(int, float ,BOOL等),所以RAC綁定一個基本類型的值是沒有問題的

 

  · RACObserve(TARGET, KEYPATH)

  作用是觀察TARGET的KEYPATH屬性,相當於KVO,產生一個RACSignal

  最常用的使用,和RAC宏綁定屬性:

RAC(self.outputLabel, text) = RACObserve(self.model, name);

  上面的代碼將label的輸出和model的name屬性綁定,實現聯動,name但凡有變化都會使得label輸出

 

  · @weakify(Obj)  @strongify(Obj)

  這對宏在 RACEXTScope.h 中定義,RACFramework好像沒有默認引入,需要單獨import

  他們的作用主要是在block內部管理對self的引用:

    @weakify(self); // 定義了一個__weak的self_weak_變量
    [RACObserve(self, name) subscribeNext:^(NSString *name) {
        @strongify(self); // 局域定義了一個__strong的self指針指向self_weak
        self.outputLabel.text = name;
    }];

  這個宏為什么這么吊,前面加@,其實就是一個啥都沒干的@autoreleasepool {}前面的那個@,為了顯眼罷了。

  這兩個宏一定成對出現,先weak再strong

 

  除了RAC中常用宏的使用,有一些宏的實現方法也很值得觀摩。 


 

  舉個高級點的栗子

  要干的一件事,計算一個可變參數列表的長度。

  第一反應就是用參數列表的api,va_start va_arg va_end遍歷一遍計算個和,但仔細想想,對於可變參數這個事,在編譯前其實就已經確定了,代碼里括號里有多少個參數一目了然。

  RAC中Racmetamarcos.h中就有一系列宏來完成這件事,硬是在預處理之后就拿到了可變參數個數:

#define metamacro_argcount(...) \
    metamacro_at(20, __VA_ARGS__, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

   這個宏由幾個工具宏一層層展開,現在模擬一下展開過程:

假如我們要計算的如下:

int count = metamacro_argcount(a, b, c);

於是乎第一層展開后:

int count = metamacro_at(20, a, b, c, 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(N, ...) metamacro_concat(metamacro_at, N)(__VA_ARGS__)
// 下面是metamacro_concat做的事(簡寫一層)
#define metamacro_concat_(A, B) A ## B

  於是乎第二層展開后:

int count = metamacro_at20(a, b, c, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1);

  再看metamacro_at20這個宏干的事兒:

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

  於是乎第三層展開后,相當於截斷了前20個參數,留下剩下幾個:

int count = metamacro_head(3, 2, 1);

  這個metamacro_head:

#define metamacro_head(...) metamacro_head_(__VA_ARGS__, 0)
#define metamacro_head_(FIRST, ...) FIRST

  后面加個0,然后取參數列表第一個,於是乎:

int count = 3;

  大功告成。

  反正我看完之后感覺挺震驚,宏還能這么用,這樣帶來的好處不止是將計算在預處理時搞定,不拖延到運行時惡心cpu;但更重要的是編譯檢查。比如某些可變參數的實現要求可以填2個參數,可以填3個參數,其他的都不行,這樣,也只有這樣的宏的實現,才能在編譯前就確定了錯誤。

 


 

   除了上面,還有一個神奇的宏的使用:

  當使用諸如RAC(self, outputLabel)或RACObserve(self, name)時,發現寫完逗號之后,輸入第二個property的時候會出現完全正確的代碼提示!這相當神奇。

  

探究一下,關鍵的關鍵是如下一個宏:

#define keypath(...) \
    metamacro_if_eq(1, metamacro_argcount(__VA_ARGS__))(keypath1(__VA_ARGS__))(keypath2(__VA_ARGS__))

  這個metamacro_argcount上面說過,是計算可變參數個數,所以metamacro_if_eq的作用就是判斷參數個數,如果個數是1就執行后面的keypath1,若不是1就執行keypath2。

  所以重點說一下keypath2:

#define keypath2(OBJ, PATH) \
    (((void)(NO && ((void)OBJ.PATH, NO)), # PATH))

  乍一看真挺懵,先化簡,由於Objc里面keypath是諸如"outputLabel.text"的字符串,所以這個宏的返回值應該是個字符串,可以簡化成:

#define keypath2(OBJ, PATH) (???????, # PATH)

  先不管"??????"是啥,這里不得不說C語言中一個不大常見的語法(第一個忽略):

int a = 0, b = 0;
a = 1, b = 2;
int c = (a, b);

  這些都是逗號表達式的合理用法,第三個最不常用了,c將被b賦值,而a是一個未使用的值,編譯器會給出warning。

  去除warning的方法很簡單,強轉成void就行了:

int c = ((void)a, b);

  再看上面簡化的keypath2宏,返回的就是PATH的字符串字面值了(單#號會將傳入值轉成字面字符串)

  (((void)(NO && ((void)OBJ.PATH, NO)), # PATH))

  綠色字體部分對傳入的第一個參數OBJ和第二個正要輸入的PATH做了點操作,這也正是為什么輸入第二個參數時編輯器會給出正確的代碼提示。強轉void就像上面說的去除了warning。

  但至於為什么加入紅色字體部分的與NO做&&,我不太能理解,我測試時其實沒有紅色部分已經完成了功能,可能是作者為了屏蔽某些隱藏的問題吧。

  這個宏的巧妙的地方就在於使得編譯器以為我們要輸入“點”出來的屬性,保證了輸入值的合法性(輸了不存在的property直接報錯的),同時利用了逗號表達式取逗號最后值的語法返回了正確的keypath。

 

總之,RAC對宏的使用達到了很高的水平,還有諸如RACTuplePack,RACTupleUnpack的宏就不細說了,值得研究。

   PS:上面介紹的metamacro和@strongify等宏確切來說來自RAC依賴的extobjc,作者是Justin Spahr-Summers,正是RAC作者之一。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM