这里写目录标题
一、概述
int getopt(int argc, char * const argv[], const char *optstring);
extern char *optarg; //这个最常用,保存一个选项的参数
extern int optind, opterr, optopt;
用法:getopt每次调用会返回main的输入中带“-”的参数。参数argc和argv是由main()传递的参数个数和内容,最后一个参数为选项字符串。这个字符串规定了选项中字符串的内容是否要带参数。
网上看的命令行组成各部分名称如下图:
很清楚哈,但定睛一看,对于getopt函数而言,其中的operands是干扰信息啊。
二、选项字符串规则:
1、 字符代表一个选项
2、 没有冒号表述纯选项,这样的参数可以连写
3、 字符后面有冒号代表选项要有参数(字符和参数可以隔开一个空格,也可不隔开)
4、 两个冒号代表选项之后的参数可写可不写,且参数与值之间不能有空格。
5、 参数使用时部分先后顺序
6、 默认情况下getopt会重新排列命令行参数的顺序,所以到最后所有不包含选项的命令行参数都排到最后。
违反规则后的后果:
如果本身字符后无冒号但使用时却带上参数:
1、 不加空格带上参数:都解析为选项了。
2、 加上空格带上参数:只解析出选项,忽略参数。
如果本身字符后一个冒号但使用时却没参数或带好多参数:
1、 多加参数,只解析第一个,其余的忽略
2、 不加参数,解析失败
参数为可选参数却有空格,则解析错误。???
比如,"vha:b:c::"
就表示:
支持-v选项,通常用于打印版本号;
支持-h选项,通常用于打印帮助信息;
支持-a选项,后面必须带有一个参数;
支持-b选项,后面必须带有一个参数;
支持-c选项,后面可以带参数,也可以不用带参数;
三、getopt 返回值
正常调用时,会返回对应的合法参数字符. 当遇到不合法的参数字符时(不包含在optstring 中),会返回’?’ . 当所有的命令行条目被解析完成后,getopt 返回-1
四、会用到的全局变量:
optarg:用来保存选项的参数(最长用)
optind:初始值为1,指明下一个待处理的元素在argv中的索引。
opterr:用来表示是否将错误信息输出到stderr,如果不希望getopt()打印出错信息,则只要将全域变量opterr设为0即可。
optopt:表示不在选项字符串optstring中的选项;????
三、示例代码
#include <stdio.h>
#include <unistd.h>
int main(int argc, char *argv[])
{
int opt;
printf("begin: optind: %d, opterr: %d \n", optind, opterr);
while((opt = getopt(argc, argv, "vha:b:c::")) != -1) {
printf("begin to prase pramater: %d \n ", optind); //optind record the position of next pramater
switch(opt) {
case 'v':
printf("version: v1.1 \n");
break;
case 'h':
printf("Usage: ./getopt_test -a 1 -b 2 -c [hello] \n");
break;
case 'a':
printf("pramater of -a: %s \n", optarg);
break;
case 'b':
printf("pramater of -b: %s \n", optarg);
break;
case 'c':
printf("pramater of -c: %s \n", optarg);
break;
case '?':
printf("pramater illegal : %s \n", optarg);
break;
default:
break;
}
}
}
四、上机实验
4.1 对选项字符串规则理解实验
4.1.1 正常情况
其中,opterr为1,表示会主动打印出错信息。
4.1.2 选项字符串v后无冒号却带参数(之间有空格)
可见,直接将参数抛弃了。
4.1.3 选项字符串v后无冒号却带参数(之间无空格)
可见他以为我输入的都是不带参数的选项。
4.1.4 选项字符串a后必须有参数却不加参数
可见他没找到参数,立马报错了。
4.1.5 选项字符串a后有冒号却有两个参数
可见多余的参数直接被忽略了
4.1.6 选项字符串c后为可选参数但却有空格
可见,可选参数与选项之间必须不能带空格。
4.2 对 optind 理解实验
举个小例子,还是上面的代码,在while循环结束后增加一条打印:“printf("optind = %d \n", optind);”提醒 getopt(argc, argv, "vha:b:c::")
",其中vh不需要带参数,ab必须带参数,c为可选参数,如果带了参数则不能加空格。
咱分析下optind 的变化过程,提示一下,optind始终指向下一个待处理的元素在argv中的索引。初始值为1:
首先optind值为1,对应-v,然后执行getopt函数时发现该元素是目标option,得让optind加1,所以打印出2。然后知道while第一遍循环结束。
开始第二遍while循环,optind值为2,对应-a,执行getopt函数时发现该元素是目标option,且需要参数,得让optind加2,所以打印出4。
开始第三遍while循环,optind值为4,对应-b,执行getopt函数时发现该元素是目标option,且需要参数,得让optind加2,所以打印出6。
开始第四遍while循环,optind值为6,对应-c,执行getopt函数时发现该元素是目标option,可选参数,但是后面有空格,当没参数处理,optind加1,所以打印出7。
开始第五遍执行while循环,发现后面没有目标option了,然后getopt返回-1,退出循环。option依然指向7。
看如果有人闲的没事干,把最后那仨option不需要的参数给放前面了会怎样?
首先optind值为1,对应hello1,然后执行getopt函数时发现该元素不是目标option,得让optind加1,getopt找到了目标参数v,optind加1指向下一个参数,所以打印出3。然后直到while第一遍循环结束。
开始第二遍while循环,getopt先重新排序argv数组(将非目标option的参数剪切插入到optind指向的参数前面),optind值为3,对应hello2,执行getopt函数时发现该元素不是目标option,得让optind加1,getopt找到了目标参数a,a带参数得让optind加2,所以打印出6。
开始第三遍while循环,getopt先重新排序argv数组(将非目标option的参数剪切插入到optind指向的参数前面)optind值为6,对应-b,执行getopt函数时发现该元素是目标option,且需要参数,得让optind加2,所以打印出8。
开始第四遍while循环,getopt先重新排序argv数组(将非目标option的参数剪切插入到optind指向的参数前面)optind值为8,对应c,执行getopt函数时发现该元素是目标option,可选参数,但是后面有空格,当没参数处理,optind加1,所以打印出9。
开始第五遍执行while循环,getopt先重新排序argv数组(将非目标option的参数剪切插入到optind指向的参数前面),发现后面没有目标option了,然后getopt返回-1,退出循环。option指向7。
发现了没有,退出while循环后option的值都是一样的。
参考链接
https://blog.csdn.net/astrotycoon/article/details/46047449