iOS 黑魔法 __attribute__((cleanup))

SD onExit 宏分析

在讀 SDWebimage 源碼的時(shí)候水由,發(fā)現(xiàn)有這么一個(gè)騷操作

 @onExit {
        if (input_buffer.data) free(input_buffer.data);
        if (output_buffer.data) free(output_buffer.data);
    };

然后就來看看 OnExit 的宏是啥樣的

#ifndef onExit
#define onExit \
sd_keywordify \
__strong sd_cleanupBlock_t metamacro_concat(sd_exitBlock_, __LINE__) __attribute__((cleanup(sd_executeCleanupBlock), unused)) = ^
#endif

typedef void (^sd_cleanupBlock_t)(void);

#if defined(__cplusplus)
extern "C" {
#endif
    void sd_executeCleanupBlock (__strong sd_cleanupBlock_t *block);
#if defined(__cplusplus)
}
#endif

這一看,我蒙了较木,這是什么騷操作侵状,之前看過有 @weakify 這種宏定義,其實(shí)就是弱引用瘪弓,前面加個(gè) @ 可能都是來源于RAC的靈感吧垫蛆,這里這個(gè)宏的定義比較復(fù)雜,我們一點(diǎn)一點(diǎn)來解釋
腺怯,首先

__strong sd_cleanupBlock_t

typedef void (^sd_cleanupBlock_t)(void);

這個(gè)都能看懂袱饭, sd_cleanupBlock_t 是下面定義的block,然后

metamacro_concat(sd_exitBlock_, __LINE__)

這個(gè)又是設(shè)么意思呢呛占,其實(shí)也是 SD 定義的一個(gè)宏

#define metamacro_concat(A, B) \
        metamacro_concat_(A, B)

#define metamacro_concat_(A, B) A ## B

看到這里是不是就明白了虑乖?就是連接兩個(gè)字符串,所以

__strong sd_cleanupBlock_t metamacro_concat(sd_exitBlock_, __LINE__) 這一行就等于

__strong sd_cleanupBlock_t sd_exitBlock_ 10 ,其實(shí)就是根據(jù)當(dāng)前行數(shù)晾虑,定義了一個(gè) block 的名字,接下來重點(diǎn)來了

__attribute__((cleanup(sd_executeCleanupBlock), unused)) = ^

attribute((cleanup)) 黑魔法

關(guān)于 cleanup 的黑魔法疹味,可以參考這篇文章
,和這篇文章我就直接將 sunnyxx 大神的文章搬過來了仅叫,做個(gè)記錄

attribute((cleanup(...))),用于修飾一個(gè)變量糙捺,在它的作用域結(jié)束時(shí)可以自動執(zhí)行一個(gè)指定的方法诫咱,如:

// 指定一個(gè)cleanup方法,注意入?yún)⑹撬揎椬兞康牡刂泛榈疲愋鸵粯?// 對于指向objc對象的指針(id *)坎缭,如果不強(qiáng)制聲明__strong默認(rèn)是__autoreleasing,造成類型不匹配
static void stringCleanUp(__strong NSString **string) {
    NSLog(@"%@", *string);
}
// 在某個(gè)方法中:
{
    __strong NSString *string __attribute__((cleanup(stringCleanUp))) = @"sunnyxx";
} // 當(dāng)運(yùn)行到這個(gè)作用域結(jié)束時(shí)签钩,自動調(diào)用stringCleanUp

所謂作用域結(jié)束掏呼,包括大括號結(jié)束、return铅檩、goto憎夷、break、exception等各種情況昧旨。
當(dāng)然拾给,可以修飾的變量不止NSString,自定義Class或基本類型都是可以的:

// 自定義的Class
static void sarkCleanUp(__strong Sark **sark) {
    NSLog(@"%@", *sark);
}
__strong Sark *sark __attribute__((cleanup(sarkCleanUp))) = [Sark new];
// 基本類型
static void intCleanUp(NSInteger *integer) {
    NSLog(@"%d", *integer);
}
NSInteger integer __attribute__((cleanup(intCleanUp))) = 1;

假如一個(gè)作用域內(nèi)有若干個(gè)cleanup的變量臼予,他們的調(diào)用順序是先入后出的棧式順序鸣戴;
而且,cleanup是先于這個(gè)對象的dealloc調(diào)用的粘拾。

既然attribute((cleanup(...)))可以用來修飾變量窄锅,block當(dāng)然也是其中之一,寫一個(gè)block的cleanup函數(shù)非常有趣:

// void(^block)(void)的指針是void(^*block)(void)
static void blockCleanUp(__strong void(^*block)(void)) {
    (*block)();
}

于是在一個(gè)作用域里聲明一個(gè)block:

{
   // 加了個(gè)`unused`的attribute用來消除`unused variable`的warning
    __strong void(^block)(void) __attribute__((cleanup(blockCleanUp), unused)) = ^{
        NSLog(@"I'm dying...");
    };
} // 這里輸出"I'm dying..."
#define onExit\
    __strong void(^block)(void) __attribute__((cleanup(blockCleanUp), unused)) = ^

用這個(gè)宏就能將一段寫在前面的代碼最后執(zhí)行:

{
    onExit {
        NSLog(@"yo");
    };
} // Log

這樣的寫法可以將成對出現(xiàn)的代碼寫在一起缰雇,比如說一個(gè)lock:

NSRecursiveLock *aLock = [[NSRecursiveLock alloc] init];
[aLock lock];
// 這里
//     有
//        100多萬行
[aLock unlock]; // 看到這兒的時(shí)候早忘了和哪個(gè)lock對應(yīng)著了

用了onExit之后入偷,代碼更集中了:

NSRecursiveLock *aLock = [[NSRecursiveLock alloc] init];
[aLock lock];
onExit {
    [aLock unlock]; // 媽媽再也不用擔(dān)心我忘寫后半段了
};
// 這里
//    愛多少行
//           就多少行

媽耶,看到這里械哟,是不是已經(jīng)豁然開朗了疏之。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市暇咆,隨后出現(xiàn)的幾起案子锋爪,更是在濱河造成了極大的恐慌,老刑警劉巖爸业,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件其骄,死亡現(xiàn)場離奇詭異,居然都是意外死亡扯旷,警方通過查閱死者的電腦和手機(jī)拯爽,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來钧忽,“玉大人毯炮,你說我怎么就攤上這事逼肯。” “怎么了桃煎?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵篮幢,是天一觀的道長。 經(jīng)常有香客問我为迈,道長迄损,這世上最難降的妖魔是什么瑞驱? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任耗啦,我火速辦了婚禮惊豺,結(jié)果婚禮上敬锐,老公的妹妹穿的比我還像新娘袜啃。我一直安慰自己盏袄,他們只是感情好芳绩,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布蛾绎。 她就那樣靜靜地躺著昆箕,像睡著了一般。 火紅的嫁衣襯著肌膚如雪租冠。 梳的紋絲不亂的頭發(fā)上鹏倘,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天,我揣著相機(jī)與錄音顽爹,去河邊找鬼纤泵。 笑死,一個(gè)胖子當(dāng)著我的面吹牛镜粤,可吹牛的內(nèi)容都是我干的捏题。 我是一名探鬼主播,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼肉渴,長吁一口氣:“原來是場噩夢啊……” “哼公荧!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起同规,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤循狰,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后券勺,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體绪钥,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年朱灿,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了昧识。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,137評論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡盗扒,死狀恐怖跪楞,靈堂內(nèi)的尸體忽然破棺而出缀去,到底是詐尸還是另有隱情,我是刑警寧澤甸祭,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布缕碎,位于F島的核電站,受9級特大地震影響池户,放射性物質(zhì)發(fā)生泄漏咏雌。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一校焦、第九天 我趴在偏房一處隱蔽的房頂上張望赊抖。 院中可真熱鬧,春花似錦寨典、人聲如沸氛雪。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽报亩。三九已至,卻和暖如春井氢,著一層夾襖步出監(jiān)牢的瞬間弦追,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工花竞, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留劲件,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓左胞,卻偏偏與公主長得像寇仓,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子烤宙,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評論 2 345