• Linux中getopt函数、optind等变量使用详解


    getopt函数、optind等变量使用详解

    最近在学习《Unix网络编程》vol2时,发现书中例子经常使用一个命令行解析getopt函数,因为函数声明比较特别,根据自己摸索,遂总结出使用方法。

    1. getopt函数的声明

    该函数是由Unix标准库提供的函数,查看命令man 3 getopt

    1. #include
    2. int getopt(int argc, char * const argv[], const char *optstring);
    3. extern char *optarg;
    4. extern int optind, opterr, optopt;

    getopt函数的参数:

    • 参数argc和argv:通常是从main的参数直接传递而来,argc是参数的数量,argv是一个常量字符串数组的地址。
    • 参数optstring:一个包含正确选项字符的字符串,如果一个字符后面有冒号,那么这个选项在传递参数时就需要跟着一个参数。

    外部变量:

    • char *optarg:如果有参数,则包含当前选项参数字符串
    • int optind:argv的当前索引值。当getopt函数在while循环中使用时,剩下的字符串为操作数,下标从optind到argc-1。
    • int opterr:这个变量非零时,getopt()函数为“无效选项”和“缺少参数选项,并输出其错误信息。
    • int optopt:当发现无效选项字符之时,getopt()函数或返回 \’ ? \’ 字符,或返回字符 \’ : \’ ,并且optopt包含了所发现的无效选项字符。

    根据下面的程序理解这四个参数

    2. getopt函数的使用

    man 3 getopt给出了该函数的例子

    1. #include
    2. #include
    3. #include
    4. #include
    5. int main(int argc, char *argv[])
    6. {
    7. int flags = 0, opt;
    8. int nsecs = 0, tfnd = 0;
    9. while((opt = getopt(argc, argv, "nt:")) != -1) {
    10. switch (opt) {
    11. case 'n':
    12. flags = 1;
    13. break;
    14. case 't':
    15. nsecs = atoi(optarg);
    16. tfnd = 1;
    17. break;
    18. default:
    19. printf("optopt = %c\n", (char)optopt);
    20. printf("opterr = %d\n", opterr);
    21. fprintf(stderr, "usage: %s [-t nsecs] [-n] name\n", argv[0]);
    22. exit(EXIT_FAILURE);
    23. }
    24. }
    25. printf("flags = %d; tfnd = %d; nsecs = %d; optind = %d\n", flags, tfnd, nsecs, optind);
    26. printf("optind = %d\n", optind);
    27. printf("argc = %d\n", argc);
    28. #if 1
    29. if(optind >= argc) {
    30. fprintf(stderr, "Expected argument after options\n");
    31. exit(1);
    32. }
    33. #endif
    34. printf("name argument = %s\n", argv[optind]);
    35. /* Other code omitted */
    36. return 0;
    37. }
    • getopt函数的第三个字符串参数 “nt:”:在man page中这样解释:-n, with no associated value; and -t val, which expects an associated value。中文意思为:-n 不用一个关联值;-t 需要有一个关联的value参数

    运行结果:

    1. test ./a.out -a name //首先 -a 是一个非法的选项
    2. ./a.out: invalid option -- 'a' //输出错误信息,因为非法选项
    3. optind = 2 //此时argv[optind]是我们的操作数,也就是我们传递给主函数的参数
    4. optopt = a //当发现无效项字符时,optopt会包含该字符,正如我们传递的‘a’这个无效项。
    5. opterr = 1 //opterr变量非零,getopt()函数为“无效选项”和“缺少参数选项,并输出其错误信息。
    6. usage: ./a.out [-t nsecs] [-n] name

    再次运行:

    1. test ./a.out -t //根据getopt函数的第三个参数,因为‘t’字符后有‘:’冒号,因此在‘-t’选项后需要跟随一个参数。因此此次运行的错误为“缺少参数”,如下提示:
    2. ./a.out: option requires an argument -- 't' //输出了错误信息,因为“缺少参数”
    3. optind = 2 //argv[optind]为空,因为此次运行没有传递参数
    4. optopt = t //当发现无效项字符时,optopt会包含该字符,正如我们传递的‘t’缺少参数的选项。
    5. opterr = 1 //opterr变量非零,getopt()函数为“无效选项”和“缺少参数“选项,并输出其错误信息。
    6. usage: ./a.out [-t nsecs] [-n] name

    再次运行:

    1. test ./a.out -t 123 -n //此时,根据字符串”-nt:“应该得知,-t 后应加参数,-n不用加
    2. flags = 1; tfnd = 1; nsecs = 123; optind = 4 //在switch语句中将flags=1,tfnd=1,因为optarg变量保存这当前选项参数的字符串,因此此时optarg保存的是‘-t’参数的参数并用atoi函数转称整数,因此nsecs=123
    3. optind = 4 //argv[optind]为空,因为没有向主函数传递参数
    4. argc = 4 //一共四个字符串
    5. Expected argument after options //该程序希望我们至少传递一个参数,因此在if语句中退出。

    再次运行:

    1. test ./a.out -t 123 -n hello //‘-t’后加参数,‘-n’不用加,向main函数传递参数叫hello
    2. flags = 1; tfnd = 1; nsecs = 123; optind = 4
    3. optind = 4 //argv[optind] = ”hello“
    4. argc = 5 //五个字符串
    5. name argument = hello //打印参数name argument= hello

    其实可以传递多个参数(如果需要,因为此函数只打印第一个参数),例如:

    1. test ./a.out -t 123 -n hello world hello C
    2. optind = 3
    3. optind = 4
    4. flags = 1; tfnd = 1; nsecs = 123; optind = 4
    5. optind = 4
    6. argc = 8
    7. name argument = hello
  • 相关阅读:
    git分支详解——记住这些指令能帮助你解决大部分git的分支问题
    firefox切换本地服务和全球服务的方法
    云安全之HTTP协议介绍补充
    后端接口性能优化分析-多线程优化
    2023年(第六届)电力机器人应用与创新发展论坛-核心PPT资料下载
    Nobe.js的安装与配置
    35 岁失业了真的会很难找到工作吗?
    Lepton 无损压缩原理及性能分析
    有人知道这个动作模型是哪里的吗?叫什么?
    24 行为型模式-访问者模式
  • 原文地址:https://blog.csdn.net/u012903992/article/details/133581665