C语言程序主要通过 main 函数的参数来传递命令行参数:
- // 默认传递命令行参数
- int main(int argc ,char * argv[])
- {
- ...
- }
其中 argc 表示参数个数(包含程序本身),argv是保存所有这些参数的二级数组。
我们可以直接操作这两个变量来处理参数,但是C标准库提供了一个更丰富的处理参数的方法,就是 getopt()。
函数声明如下:
- #include
-
- int getopt(int argc, char * const argv[], const char *optstring);
getopt() 函数用于解析命令行参数。它的 argc 和 argv 参数通常直接从 main() 的参数直接传递而来。
optstring 是可以处理选项字母组成的字符串。该字串里的每个字符对应于一个以 ‘-’ 开头的选项。如果该字串里的任一字符后面有冒号,那么这个选项就要求有参数(如“hd:”对应于 '-h' 和 '-d', 其中 '-d' 后需接参数)。而如果选项后面接两个冒号,则说明这个选项后的参数是可选的,即可带参数也可不带参数。
当给定 getopt() 命令参数的数量 (argc)、指向这些参数的数组 (argv) 和选项字串 (optstring) 后,getopt() 将返回第一个选项,并设置一些全局变量。
- extern char *optarg;
- extern int optind, opterr, optopt;
getopt() 调用时会根据 optind 和 optstring 遍历选项和参数,如果找到一个选项字符,则返回该字符,同时更新全局变量 optind,optarg 以及静态变量 nextchar,以便下一次调用 getopt() 时可以继续扫描。
如果 getopt() 被重复调用,它会依次返回每个选项字符。如果所有可识别的选项都被扫描过,将返回 -1,此时 optind 是第一个不是选项的 argv 元素的索引。
默认情况下,getopt() 会在扫描时置换 argv 的内容,以便扫描完成后所有的非选项都处于末尾。但还提供了另外两种扫描模式。
1. 一旦遇到非选项参数就立即停止,通过 optstring 的一个字符设置为 '+' 来选择这种模式;
2. 每个非选项 argv 元素都被处理为好像它是具有字符代码 1 的选项的参数一样,此时碰到那些没有选项对应的参数时不会报错也不会退出,而是认为它是选项“1”的参数。通过 optstring 的第一个字符设置为 ‘-’ 来选择这种模式。
注:无论处于那种扫描模式,碰到参数 '--' 都会强制结束选项扫描。
在处理选项列表时,getopt() 可以检测到两种错误:
1. 未在 optstring 中定义的选项字符;
2. 缺少选项参数;
默认情况下, getopt() 会打印一条错误消息,将错误的选项字符放在 optopt 中,并返回 '?' 作为函数结果。但如果我们想要其他的处理方式,可以有以下做法:
1. 在调用 getopt() 前设置 opterr 为零,此时 getopt() 不会输出错误信息,我们可以通过返回值是否为 '?' 来进行不同的处理。
2. 将 optstring 的第一个字符设置为 ‘+’ 或者 '-',此时 getopt() 同样不会输出错误信息,同时可以根据返回值是 ‘?’ 还是 ':' 来做不同的处理。 ‘?’ 表示未定义选项,':' 表示缺少选项参数
以下简单的示例程序使用 getopt() 处理两个程序选项: -n,没有关联值;和 -t val,它需要一个关联的值。
- #include
- #include
- #include
-
- int main(int argc, char *argv[])
- {
- int flags, opt;
- int nsecs, tfnd;
-
- nsecs = 0;
- tfnd = 0;
- flags = 0;
- while ((opt = getopt(argc, argv, "nt:")) != -1) {
- switch (opt) {
- case 'n':
- flags = 1;
- break;
- case 't':
- nsecs = atoi(optarg);
- tfnd = 1;
- break;
- default: /* '?' */
- fprintf(stderr, "Usage: %s [-t nsecs] [-n] name\n",
- argv[0]);
- exit(EXIT_FAILURE);
- }
- }
-
- printf("flags=%d; tfnd=%d; nsecs=%d; optind=%d\n",
- flags, tfnd, nsecs, optind);
-
- if (optind >= argc) {
- fprintf(stderr, "Expected argument after options\n");
- exit(EXIT_FAILURE);
- }
-
- printf("name argument = %s\n", argv[optind]);
-
- exit(EXIT_SUCCESS);
- }
getopt函数_sdoyuxuan的博客-CSDN博客_getopt函数
man entry of getopt