練習(xí)25:變參函數(shù)
譯者:飛龍
在C語言中缭受,你可以通過創(chuàng)建“變參函數(shù)”來創(chuàng)建你自己的printf
或者scanf
版本蠕啄。這些函數(shù)使用stdarg.h
頭,它們可以讓你為你的庫創(chuàng)建更加便利的接口舟奠。它們對于創(chuàng)建特定類型的“構(gòu)建”函數(shù)胆剧、格式化函數(shù)和任何用到可變參數(shù)的函數(shù)都非常實(shí)用骇塘。
理解“變參函數(shù)”對于C語言編程并不必要汛闸,我在編程生涯中也只有大約20次用到它。但是平痰,理解變參函數(shù)如何工作有助于你對它的調(diào)試汞舱,并且讓你更加了解計(jì)算機(jī)。
/** WARNING: This code is fresh and potentially isn't correct yet. */
#include <stdlib.h>
#include <stdio.h>
#include <stdarg.h>
#include "dbg.h"
#define MAX_DATA 100
int read_string(char **out_string, int max_buffer)
{
*out_string = calloc(1, max_buffer + 1);
check_mem(*out_string);
char *result = fgets(*out_string, max_buffer, stdin);
check(result != NULL, "Input error.");
return 0;
error:
if(*out_string) free(*out_string);
*out_string = NULL;
return -1;
}
int read_int(int *out_int)
{
char *input = NULL;
int rc = read_string(&input, MAX_DATA);
check(rc == 0, "Failed to read number.");
*out_int = atoi(input);
free(input);
return 0;
error:
if(input) free(input);
return -1;
}
int read_scan(const char *fmt, ...)
{
int i = 0;
int rc = 0;
int *out_int = NULL;
char *out_char = NULL;
char **out_string = NULL;
int max_buffer = 0;
va_list argp;
va_start(argp, fmt);
for(i = 0; fmt[i] != '\0'; i++) {
if(fmt[i] == '%') {
i++;
switch(fmt[i]) {
case '\0':
sentinel("Invalid format, you ended with %%.");
break;
case 'd':
out_int = va_arg(argp, int *);
rc = read_int(out_int);
check(rc == 0, "Failed to read int.");
break;
case 'c':
out_char = va_arg(argp, char *);
*out_char = fgetc(stdin);
break;
case 's':
max_buffer = va_arg(argp, int);
out_string = va_arg(argp, char **);
rc = read_string(out_string, max_buffer);
check(rc == 0, "Failed to read string.");
break;
default:
sentinel("Invalid format.");
}
} else {
fgetc(stdin);
}
check(!feof(stdin) && !ferror(stdin), "Input error.");
}
va_end(argp);
return 0;
error:
va_end(argp);
return -1;
}
int main(int argc, char *argv[])
{
char *first_name = NULL;
char initial = ' ';
char *last_name = NULL;
int age = 0;
printf("What's your first name? ");
int rc = read_scan("%s", MAX_DATA, &first_name);
check(rc == 0, "Failed first name.");
printf("What's your initial? ");
rc = read_scan("%c\n", &initial);
check(rc == 0, "Failed initial.");
printf("What's your last name? ");
rc = read_scan("%s", MAX_DATA, &last_name);
check(rc == 0, "Failed last name.");
printf("How old are you? ");
rc = read_scan("%d", &age);
printf("---- RESULTS ----\n");
printf("First Name: %s", first_name);
printf("Initial: '%c'\n", initial);
printf("Last Name: %s", last_name);
printf("Age: %d\n", age);
free(first_name);
free(last_name);
return 0;
error:
return -1;
}
這個(gè)程序和上一個(gè)練習(xí)很像宗雇,除了我編寫了自己的scanf
風(fēng)格函數(shù)昂芜,它以我自己的方式處理字符串。你應(yīng)該對main
函數(shù)很清楚了赔蒲,以及read_string
和read_int
兩個(gè)函數(shù)泌神,因?yàn)樗鼈儾]有做什么新的東西。
這里的變參函數(shù)叫做read_scan
舞虱,它使用了va_list
數(shù)據(jù)結(jié)構(gòu)執(zhí)行和scanf
相同的工作欢际,并支持宏和函數(shù)。下面是它的工作原理:
- 我將函數(shù)的最后一個(gè)參數(shù)設(shè)置為
...
矾兜,它向C表示這個(gè)函數(shù)在fmt
參數(shù)之后接受任何數(shù)量的參數(shù)损趋。我可以在它前面設(shè)置許多其它的參數(shù),但是在它后面不能放置任何參數(shù)椅寺。 - 在設(shè)置完一些參數(shù)時(shí)浑槽,我創(chuàng)建了
va_list
類型的變量墙杯,并且使用va_list
來為其初始化。這配置了stdarg.h
中的這一可以處理可變參數(shù)的組件括荡。 - 接著我使用了
for
循環(huán),遍歷fmt
格式化字符串溉旋,并且處理了類似scanf
的格式畸冲,但比它略簡單。它里面只帶有整數(shù)观腊、字符和字符串邑闲。 - 當(dāng)我碰到占位符時(shí),我使用了
switch
語句來確定需要做什么梧油。 - 現(xiàn)在苫耸,為了從
va_list argp
中獲得遍歷,我需要使用va_arg(argp, TYPE)
宏儡陨,其中TYPE
是我將要向參數(shù)傳遞的準(zhǔn)確類型褪子。這一設(shè)計(jì)的后果是你會非常盲目,所以如果你沒有足夠的變量傳入骗村,程序就會崩潰嫌褪。 - 和
scanf
的有趣的不同點(diǎn)是,當(dāng)它碰到's'
占位符時(shí)胚股,我使用read_string
來創(chuàng)建字符串笼痛。va_list argp
棧需要接受兩個(gè)函數(shù):需要讀取的最大尺寸,以及用于輸出的字符串指針琅拌。read_string
使用這些信息來執(zhí)行實(shí)際工作缨伊。 - 這使
read_scan
比scan
更加一致,因?yàn)槟憧偸鞘褂?code>&提供變量的地址进宝,并且合理地設(shè)置它們刻坊。 - 最后,如果它碰到了不在格式中的字符党晋,它僅僅會讀取并跳過紧唱,而并不關(guān)心字符是什么,因?yàn)樗恍枰^隶校。
你會看到什么
當(dāng)你運(yùn)行程序時(shí)漏益,會得到與下面詳細(xì)的結(jié)果:
$ make ex25
cc -Wall -g -DNDEBUG ex25.c -o ex25
$ ./ex25
What's your first name? Zed
What's your initial? A
What's your last name? Shaw
How old are you? 37
---- RESULTS ----
First Name: Zed
Initial: 'A'
Last Name: Shaw
Age: 37
如何使它崩潰
這個(gè)程序?qū)彌_區(qū)溢出更加健壯,但是和scanf
一樣深胳,它不能夠處理輸入的格式錯(cuò)誤绰疤。為了使它崩潰,試著修改代碼舞终,把首先傳入用于'%s'
格式的尺寸去掉轻庆。同時(shí)試著傳入多于MAX_DATA
的數(shù)據(jù)癣猾,之后找到在read_string
中不使用calloc
的方法,并且修改它的工作方式余爆。最后還有個(gè)問題是fgets
會吃掉換行符纷宇,所以試著使用fgetc
修復(fù)它,要注意字符串結(jié)尾應(yīng)為'\0'
蛾方。
附加題
- 再三檢查確保你明白了每個(gè)
out_
變量的作用像捶。最重要的是out_string
,并且它是指針的指針桩砰。所以拓春,理清當(dāng)你設(shè)置時(shí)獲取到的是指針還是內(nèi)容尤為重要。 - 使用變參系統(tǒng)編寫一個(gè)和
printf
相似的函數(shù)亚隅,重新編寫main
來使用它硼莽。 - 像往常一樣,閱讀這些函數(shù)/宏的手冊頁煮纵,確保知道了它在你的平臺做了什么懂鸵,一些平臺會使用宏而其它平臺會使用函數(shù),還有一些平臺會讓它們不起作用行疏。這完全取決于你所用的編譯器和平臺矾瑰。