RAC宏分析8:@keypath

Created by 大劉 liuxing8807@126.com

參考:
https://juejin.im/post/58a0781861ff4b006b4cfe30
http://lldong.github.io/2014/02/24/key-paths-validation.html
https://onevcat.com/2014/01/black-magic-in-macro/

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

#define keypath1(PATH) \
(((void)(NO && ((void)PATH, NO)), strchr(# PATH, '.') + 1))

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

假設(shè)我們這樣寫:

NSString *versionPath = @keypath(NSObject, version);
NSLog(@"%@", versionPath); // version

我們來(lái)分析一下牵啦,為什么@keypath(NSObject, version)就變成了字符串@"version"
由于我們傳了兩個(gè)參數(shù):NSObject & version, 所以會(huì)走宏:

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

這里面OBJ->NSObject, PATH->version

  1. 加void是為了防止逗號(hào)表達(dá)式warning, 例如:
    int a=0; int b = 1;
    int c = (a,b);
    由于a沒(méi)有被用到,所以會(huì)有警告。但是寫成如下的樣子就不會(huì)出現(xiàn)警告了:
    int c = ((void)a,b);
    所以上面加了幾個(gè)void就是為了防止出現(xiàn)warning.

比如我們仿照著這種寫法定義自己的宏:

#define ZZ_KEY_PATH(OBJ, PATH) \
(((NO && ((void)OBJ.PATH, NO)), # PATH))
    
    NSString *s = @ZZ_KEY_PATH(self.view, backgroundColor);
    NSLog(@"%@", s); // backgroundColor

如果在寫宏ZZ_KEY_PATH(OBJ, PATH)的時(shí)候把void去掉:

#define ZZ_KEY_PATH(OBJ, PATH) \
(((NO && (OBJ.PATH, NO)), # PATH))

程序運(yùn)行不受影響贞岭,但編譯器就會(huì)提示警告:


image.png

這就是上面的宏定義加void的原因

  1. 加NO是C語(yǔ)言判斷條件短路表達(dá)式无埃。增加NO && 以后鹦付,預(yù)編譯的時(shí)候看見(jiàn)了NO屎即,就會(huì)很快的跳過(guò)判斷條件贡翘。即:
#define keypath2(OBJ, PATH) \
(((void)(NO && ((void)OBJ.PATH, NO)), # PATH))
其最終結(jié)果相當(dāng)于:#path
但是為什么要加這個(gè)NO蚜厉,這是因?yàn)闉榱舜a提示长已,有了OBJ.PATH,由于這里的點(diǎn)昼牛,所以輸入第二個(gè)參數(shù)時(shí)編輯器會(huì)給出正確的代碼提示(關(guān)于代碼提示等下還會(huì)說(shuō))

3.) # PATH, 這個(gè)#操作符可以在預(yù)處理階段將后面的宏參數(shù)展開為一個(gè)C的字符串常量术瓮,比如:

#define ZZ_STR(s) # s
NSString *str = @ZZ_STR(我的心太亂);
NSLog(@"%@", str);
  1. @, @keypath加這個(gè)@是因?yàn)楹甓xkeypath生成了一個(gè)C字符串,加上@即OC字符串贰健,就像上面的ZZ_STR(s) #s -> @ZZ_STR(s) -> @#s -> @"s表示的字符串"

  2. 關(guān)于keypath1的strchr(# PATH, '.')

#define keypath1(PATH) \
(((void)(NO && ((void)PATH, NO)), strchr(# PATH, '.') + 1))

strchar函數(shù)是C中<string.h>中定義的函數(shù)胞四,用來(lái)查找某字符在字符串中首次出現(xiàn)的位置,其原型為:
char * strchr (const char *str, int c);
【參數(shù)】str 為字符串伶椿,c 為要查找的字符辜伟。
strchr() 將會(huì)找出 str 字符串中第一次出現(xiàn)的字符 c 的地址,然后將該地址返回悬垃。
注意:字符串 str 的結(jié)束標(biāo)志 NUL 也會(huì)被納入檢索范圍游昼,所以 str 的組后一個(gè)字符也可以被定位。
【返回值】如果找到指定的字符則返回該字符所在地址尝蠕,否則返回 NULL烘豌。
返回的地址是字符串在內(nèi)存中隨機(jī)分配的地址再加上你所搜索的字符在字符串位置。設(shè)字符在字符串中首次出現(xiàn)的位置為 i看彼,那么返回的地址可以理解為 str + i廊佩。

示例:

#include <stdio.h>
#include <string.h>

int main(int argc, const char * argv[]) {
    const char *s = "Hello world, hello china!";
    char *p = strchr(s, 'e');
    printf("%p\n", s); // 0x100000f8c
    printf("%p\n", p); // 0x100000f8d
    printf("%s\n", p); // ello world, hello china!
    return 0;
}

也就是說(shuō)囚聚,strchr(str, 'c'), 返回的是一個(gè)C字符串,這個(gè)字符串從找到str中為‘c’的字符開始往后:

#define ZZ_KEY_PATH(PATH) \
strchr(# PATH, '.')

int main(int argc, const char * argv[]) {
    printf("%s\n", strchr("Hello.china.shanghai", '.')); // .china.shanghai
    printf("%s\n", ZZ_KEY_PATH(Hello.china.shanghai)); // .china.shanghai
    return 0;
}

再來(lái)看keypath1的宏定義:

#define keypath1(PATH) \
(((void)(NO && ((void)PATH, NO)), strchr(# PATH, '.') + 1))
我們先把它簡(jiǎn)化一下為:
#define keypath1(PATH) \
strchr(# PATH, '.') + 1)

#include <stdio.h>
#include <string.h>

#define keypath1(PATH) \
strchr(# PATH, '.') + 1

int main(int argc, const char * argv[]) {
    printf("%s\n", keypath1(Hello.china.shanghai)); // china.shanghai
    return 0;
}
  1. 關(guān)于代碼提示
    為什么在輸入keypath1或keypath2宏的時(shí)候标锄,會(huì)出現(xiàn)代碼提示顽铸?并且編譯器可以檢測(cè)到是否合法。
    來(lái)看示例:
#include <stdio.h>
#include <string.h>

#define my_keypath1(PATH) \
strchr(# PATH, '.') + 1

#define keypath1(PATH) \
(((void)(NO && ((void)PATH, NO)), strchr(# PATH, '.') + 1))

int main(int argc, const char * argv[]) {
    printf("%s\n", my_keypath1(Hello.china.shanghai)); // china.shanghai
    printf("%s\n", keypath1(Hello.china.shanghai)); // 編譯不通過(guò):Use of undeclared identifier 'Hello'
    return 0;
}

keypath1編譯不通過(guò)料皇,這是因?yàn)槎禾?hào)表達(dá)式的第一項(xiàng):
((void)(NO && ((void)PATH, NO))
即谓松,如果PATH輸入的是Hello.china.shanghai, 它就會(huì)作為表達(dá)式的一部分,它不是一個(gè)合法的表達(dá)式践剂,因此編譯不通過(guò)鬼譬。
另外再說(shuō)一下代碼提示:

image111.png

之所以會(huì)提示,也是由于這個(gè)表達(dá)式的原因逊脯。只要是作為表達(dá)式的一部分优质,XCode工具自動(dòng)會(huì)提示, 這并沒(méi)有什么神奇的地方:

image222.png

END

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市军洼,隨后出現(xiàn)的幾起案子巩螃,更是在濱河造成了極大的恐慌,老刑警劉巖匕争,帶你破解...
    沈念sama閱讀 218,858評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件避乏,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡甘桑,警方通過(guò)查閱死者的電腦和手機(jī)淑际,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)扇住,“玉大人春缕,你說(shuō)我怎么就攤上這事∷姨#” “怎么了锄贼?”我有些...
    開封第一講書人閱讀 165,282評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)女阀。 經(jīng)常有香客問(wèn)我宅荤,道長(zhǎng),這世上最難降的妖魔是什么浸策? 我笑而不...
    開封第一講書人閱讀 58,842評(píng)論 1 295
  • 正文 為了忘掉前任冯键,我火速辦了婚禮,結(jié)果婚禮上庸汗,老公的妹妹穿的比我還像新娘惫确。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評(píng)論 6 392
  • 文/花漫 我一把揭開白布改化。 她就那樣靜靜地躺著掩蛤,像睡著了一般。 火紅的嫁衣襯著肌膚如雪陈肛。 梳的紋絲不亂的頭發(fā)上揍鸟,一...
    開封第一講書人閱讀 51,679評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音句旱,去河邊找鬼阳藻。 笑死,一個(gè)胖子當(dāng)著我的面吹牛谈撒,可吹牛的內(nèi)容都是我干的稚配。 我是一名探鬼主播,決...
    沈念sama閱讀 40,406評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼港华,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了午衰?” 一聲冷哼從身側(cè)響起立宜,我...
    開封第一講書人閱讀 39,311評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎臊岸,沒(méi)想到半個(gè)月后橙数,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,767評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡帅戒,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評(píng)論 3 336
  • 正文 我和宋清朗相戀三年灯帮,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片逻住。...
    茶點(diǎn)故事閱讀 40,090評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡钟哥,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出瞎访,到底是詐尸還是另有隱情腻贰,我是刑警寧澤,帶...
    沈念sama閱讀 35,785評(píng)論 5 346
  • 正文 年R本政府宣布扒秸,位于F島的核電站播演,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏伴奥。R本人自食惡果不足惜写烤,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望拾徙。 院中可真熱鬧洲炊,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至古徒,卻和暖如春拓提,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背隧膘。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評(píng)論 1 271
  • 我被黑心中介騙來(lái)泰國(guó)打工代态, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人疹吃。 一個(gè)月前我還...
    沈念sama閱讀 48,298評(píng)論 3 372
  • 正文 我出身青樓蹦疑,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親萨驶。 傳聞我的和親對(duì)象是個(gè)殘疾皇子歉摧,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容