linux 下的帮助接口argp_parse()
选项能够改变程序的运行轨迹,而帮助系统需要让用户知道有哪些选项可以使用。
gnu 的帮助系统已经为我们写了很多代码,我们只需要调用argp_parse 函数就可以了.
----------------------------------------
argp_parse() 的基本用法
----------------------------------------
argp_parse需要我们定义一个帮助表,看一下它的调用参数: 6个,挺复杂
int argp_parse (const struct argp * __argp,
int __argc, char ** __argv,
unsigned __flags, int * __arg_index,
void * __input);
第1项是一个结构指针
第2项,第3项argc,argv 就是c语言的命令行参数选项.
第4项, flag.
第5项,index, 第6项input
不明白的可以都设置为0,看看是什么情况.
- #include
- #include
- int main (int argc, char **argv)
- {
- return argp_parse (0, argc, argv, 0, 0, 0);
- }
-
argc,argv不能填0, 否则会段错误.
该程序编译运行后,果然能很好的工作,出现了默认的帮助菜单,与标准linux的显示风格一致.
现在想加入版本信息.只需要加一个全局变量即可.
const char *argp_program_version = "version 1.0";
工作的也很好!
这些都是系统的东西,我们基本上没有做什么.
注意这里所说的系统就是库,相对于用户来说的,以后都沿用这种说法.
现在我们想加入一个自己的选项, 先定义一个选项表,里面包含一个选项.
struct argp_option options[] =
{
{ "dot", 'd', "NUM", OPTION_ARG_OPTIONAL, "Show some dots on the screen"},
{ 0 },
};
选项还是容易理解的,如下定义,其中1,3,5项都是字符串,第1项是长选项名称
type = struct argp_option {
const char *name;
int key;
const char *arg;
int flags;
const char *doc;
int group;
}
第1项是长选项名称
第3项是参数提示信息,上例"NUM" 字符串会跟在长选项提示信息中.
第5项是帮助信息.
第2项是短选项,一个字符,所以用''括住,它是作为key来使用的, 各个选项是靠它来区分的.
第4项flags, OPTION_ARG_OPTIONAL 表示该参数是可选的(可以跟参数或不跟参数)
第6项group, 是分组数值,后面使用时就理解了.
那这个options 表如何使用呢?
后面,有一个重要的结构要出场了,它就是 struct argp, 有7个成员
type = struct argp {
const struct argp_option *options;
argp_parser_t parser;
const char *args_doc;
const char *doc;
const struct argp_child *children;
char *(*help_filter)(int, const char *, void *);
const char *argp_domain;
}
这个结构太复杂,要慢慢通过实例来理解. 不过不理解的可以先置空.
第一项 options 指针显然指向我们定义的argp_option 数组
第2项 parser 是一个回调函数。 你想,当我们定义了一个选项,通过调用 --help系统把帮助信息给我们显示了出来.
当我们从命令行输入自定义选项时,怎样解释这个选项,也是需要我们来解释的,所以我们要定义这个解释函数。
第3项,4项都是字符串,用用就知道了,其它都置0先不管。
- #include
- #include
- #include
- const char *argp_program_version = "version 1.0";
- struct argp_option options[] =
- {
- { "dot", 'd', "NUM", OPTION_ARG_OPTIONAL, "Show some dots on the screen"},
- { 0 },
- };
-
- //key 是命令行中传入的选项的key, 如果该key在options表中定义了参数,arg传递的是命令行中的选项参数.
- static int parse_opt (int key, char *arg, struct argp_state *state)
- {
- switch (key)
- {
- case 'd':
- {
- unsigned int i;
- unsigned int dots = 1;
- if (arg != NULL) dots = atoi (arg);
- printf("you input option d number is :%d\n",dots);
- for (i = 0; i < dots; i++) printf (".");
- printf("\n");
- break;
- }
- }
- return 0; //很关键,否则无信息提示了!!
- }
-
- struct argp argp = {options, parse_opt, "
," }; //参数3是跟在使用提示的字符串 -
- int main (int argc, char **argv)
- {
- return argp_parse (&argp, argc, argv, 0, 0, 0);
- }
-
"have a rest!", 目前只定义了一个可选参数OPTION_ARG_OPTIONAL, 类型为"NUM", 工作运行良好, 已经可以应付一般的帮助要求了.
解释一下吧,为什么设置了argp_program_version 就能显示出版本号? argp_parse 是怎样得知信息的?
解释清这个问题,需要刨老底了.
我查看了它的库代码,原来如此. (库代码请参考链接下载)
./gnu/argp-pv.c|25| const char *argp_program_version
./gnu/argp.h|448| extern const char *argp_program_version;
使用时:
if (argp_program_version)
{
....
}
就是说,库里边定义了argp_program_version变量,但它的值是NULL,如果用户也定义了argp_program_version,并且不为0,
那应用就会链接你定义的变量, 由于其不为0, 所以就可以添加版本显示选项了.
----------------------------------------
argp 的高级用法
----------------------------------------
1. 介绍2个key参数: ARGP_KEY_ARG,ARGP_KEY_END, 用来处理参数
我们知道,命令行除了可以带选项,还可以带参数,每当系统分析到参数时,就会回调parse_opt, 并传递key参数为ARGP_KEY_ARG,
当所有参数处理完时,传递ARGP_KEY_END key 参数
在parse_opt 函数中,会有如下分支
case ARGP_KEY_ARG
case ARGP_KEY_END
2. 添加分组信息,使输出帮助信息更加明了
参考下面程序:
- #include
- #include
- #include
- const char *argp_program_version = "version 1.0";
- struct argp_option options[] =
- {
- { 0, 0, 0, 0, "Program Options:", 7}, //自定义信息这里定义为7
- { "dot", 'd', "NUM", OPTION_ARG_OPTIONAL, "Show some dots on the screen"},
- { 0, 0, 0, 0, "Informational Options:", -1}, //系统help 信息属于-1分组
- { 0 },
- };
-
- static int parse_opt (int key, char *arg, struct argp_state *state)
- {
- int *arg_count = state->input;
- switch (key)
- {
- case 'd':
- {
- unsigned int i;
- unsigned int dots = 1;
- if (arg != NULL) dots = atoi (arg);
- for (i = 0; i < dots; i++) printf (".");
- printf("\n");
- break;
- }
- case ARGP_KEY_ARG:
- (*arg_count)--;
- if (*arg_count >= 0) printf (" %s", arg);
- break;
- case ARGP_KEY_END:
- printf ("\n");
- if (*arg_count >= 4) argp_failure (state, 1, 0, "too few arguments");
- else if (*arg_count < 0) argp_failure (state, 1, 0, "too many arguments");
- break;
- }
- return 0; //很关键,否则无信息提示了!!
- }
-
- struct argp argp = {options, parse_opt, "[param1 [param2 [param3 [param4]]]]" };
-
- int main (int argc, char **argv)
- {
- int arg_count = 4;
- return argp_parse (&argp, argc, argv, 0, 0, &arg_count); //第5项input, 会传递给回调函数的state->input
- }
第6个参数你可以传递一个整数,也可以传递一个结构以包含更多信息.
argp_failure 函数实际上类似于printf, 不过它有统一的,标准的外观和感觉.
先介绍到这里吧.
1.介绍了struct argp_option 及其6个成员.
2.介绍了struct argp 7个成员中的3个成员
3.介绍了argp_parse函数6个参数中的4个参数
以后有需要再扩充:
参考1: http://nongnu.askapache.com/argpbook/step-by-step-into-argp.pdf
参考2: https://blog.csdn.net/sinat_38816924/article/details/122180938#t10 (对英文版的翻译)
参考3: gnu 库代码