[iOS-Objective-C] 格式化字符串變參函數(shù)的實(shí)現(xiàn)

有些時(shí)候,可能需要通過 Objective-C 實(shí)現(xiàn)一個(gè)接收格式化字符串可變參數(shù)的函數(shù),如 Foundation 中的某些方法一樣:

FOUNDATION_EXPORT void NSLog(NSString *format, ...) NS_FORMAT_FUNCTION(1,2) NS_NO_TAIL_CALL;

- (instancetype)initWithFormat:(NSString *)format, ... NS_FORMAT_FUNCTION(1,2);

實(shí)現(xiàn)變參函數(shù)需要用到C語(yǔ)言中關(guān)于變參的一組宏:va_start、va_arg、va_end。va 是可變參數(shù) variable argument 的意思。使用方式為:

- (void)vaTestMethod:(NSInteger)unavaliable, ... {
    va_list ap;
    va_start(ap, unavaliable);
    while (YES) {
        NSString *string = va_arg(ap, NSString*);
        if (!string) {
            break;
        }
        NSLog(@"%@",string);
    }
    va_end(ap);
}

- (void)test {
    [self vaTestMethod:0, @"1", @"2", @"3", nil];
}
  1. 聲明一個(gè)va_list類型的變量,如 ap,這個(gè)變量是指向參數(shù)的指針。
  2. va_start宏初始化變量 ap,這個(gè)宏的第二個(gè)參數(shù)是可變參數(shù)的前一個(gè)固定參數(shù)。這就使得我們實(shí)現(xiàn)的函數(shù)在可變參數(shù)前,必須至少包含一個(gè)固定參數(shù)。
  3. va_arg返回可變的參數(shù),這個(gè)宏的第二個(gè)參數(shù)是你要返回的參數(shù)的類型。
  4. va_end宏結(jié)束可變參數(shù)的獲取。

從上例可以看出,這種方式的使用場(chǎng)景十分有限。首先這組宏沒有提供對(duì)參數(shù)個(gè)數(shù)的檢測(cè),只能通過在參數(shù)末尾傳入 nil,或者像NSLog函數(shù)一樣,根據(jù)第一個(gè)固定參數(shù) format 來判斷參數(shù)個(gè)數(shù)。另外,需要在函數(shù)體內(nèi)知道可變參數(shù)中的每個(gè)參數(shù)的類型,同樣需要通過固定參數(shù) format 來獲取相關(guān)信息。

所以可以通過這種方式實(shí)現(xiàn)一個(gè)接收格式化字符串可變參數(shù)的函數(shù)。比如可以在自定義 log 函數(shù)的實(shí)現(xiàn)中應(yīng)用。NSString提供了方法- initWithFormat:arguments:直接接收 format 和 ap 參數(shù)轉(zhuǎn)換成 string 對(duì)象,從而無(wú)需開發(fā)者自己根據(jù) format 判斷要獲取的參數(shù)類型和數(shù)量。

- (void)log:(NSString *)format, ... {    
    va_list ap;
    va_start(ap, format);
    NSString *information = [[NSString alloc] initWithFormat:format arguments:ap];
    va_end(ap);
    fprintf(stderr,"%s\n", [information UTF8String]);
}

va 宏原理

C語(yǔ)言的函數(shù)參數(shù)是以棧這種數(shù)據(jù)結(jié)構(gòu)來存取的,在函數(shù)參數(shù)列表中,從右至左依次入棧存入?yún)?shù)的內(nèi)存地址,我們運(yùn)行va_start(ap, v)后,ap就指向第一個(gè)可變參數(shù)在棧的地址,然后我們用va_arg(ap, t)取得類型t的可變參數(shù)值。之后 ap 就會(huì)指向這個(gè)參數(shù)后的地址。最后通過va_end使 ap 不再指向棧。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 我們?cè)贑語(yǔ)言編程中會(huì)遇到一些參數(shù)個(gè)數(shù)可變的函數(shù),例如printf() 這個(gè)函數(shù),它的定義是這樣的: int pri...
    Mr_Me閱讀 852評(píng)論 0 2
  • 代碼: -(NSString*)makeDrink:(NSString*)drink Fruit:(NSStrin...
    bada閱讀 1,895評(píng)論 0 0
  • 1.在C中,變長(zhǎng)參數(shù)的函數(shù)即參數(shù)個(gè)數(shù)可變、參數(shù)類型不定 的函數(shù)。當(dāng)我們無(wú)法列出傳遞函數(shù)的所有實(shí)參的類型和數(shù)目時(shí),以...
    孤獨(dú)雜貨鋪閱讀 1,940評(píng)論 1 1
  • 不管我們天生是什么樣子,是什么狀況,父母給我們帶來了什么。我們所遭遇的一切痛苦與快樂,其最終我們都是在自覺與...
    烽火煤閱讀 200評(píng)論 0 0
  • 曾經(jīng)意氣風(fēng)發(fā)特立獨(dú)行如我, 自由浪漫固執(zhí)感性如我 一場(chǎng)似曾相識(shí)的緣分在最不經(jīng)意時(shí)相遇 于是,美好如意料之外的命中注...
    魚與瑜閱讀 426評(píng)論 2 2

友情鏈接更多精彩內(nèi)容