getopt和getopt_long

来源:互联网 发布:centos 禁止ping 编辑:程序博客网 时间:2024/05/30 23:47

原文地址:http://www.cnblogs.com/oloroso/p/4616282.html

getopt(分析命令行参数)

  • getopt(分析命令行参数)
    • 短参数的定义
    • 返回值
    • 范例
    • getopt_long
  • 相关函数表头文件
    #include<unistd.h> 转载时注:还有另一个头文件getopt.h
  • 定义函数
    int getopt(int argc,char * const argv[ ],const char * optstring);
  • 函数说明
    getopt()用来分析命令行参数。参数argcargv是由main()传递的参数个数内容。参数optstring 则代表欲处理的选项字符串。此函数会返回在argv中下一个的选项字母,此字母会对应参数optstring 中的字母。如果选项字符串里的字母后接着冒号“:”,则表示还有相关的参数,全域变量optarg 即会指向此额外参数。如果getopt()找不到符合的参数,则会打印出错信息,并将全域变量optopt设为“?”字符,如果不希望getopt()打印出错信息,则只要将全域变量opterr设为0即可。

短参数的定义

getopt()使用optstring所指的字串作为短参数列表,象"1ac:d::"就是一个短参数列表。短参数的定义是一个-后面跟一个字母或数字,象-a,-b就是一个短参数。每个数字或字母定义一个参数。

其中短参数在getopt定义里分为三种:

  1. 不带值的参数,它的定义即是参数本身。
  2. 必须带值的参数,它的定义是在参数本身后面再加一个冒号。
  3. 可选值的参数,它的定义是在参数本身后面加两个冒号 。

在这里拿上面的"1ac:d::"作为样例进行说明,其中的1,a就是不带值的参数,c必须带值的参数,d可选值的参数。
  在实际调用中,-1 -a -c cvalue -d-1 -a -c cvalue -ddvalue,-1a -ddvalue -c cvalue都是合法的。这里需要注意三点:

  1. 不带值的参数可以连写,象1a是不带值的参数,它们可以-1 -a分开写,也可以-1a-a1连写。
  2. 参数不分先后顺序,-1a -c cvalue -ddvalue-d -c cvalue -a1的解析结果是一样的。
  3. 要注意可选值的参数的参数之间不能有空格,必须写成-ddvalue这样的格式,如果写成-d dvalue这样的格式就会解析错误。

默认情况下getopt会`重新排列命令行参数的顺序,所以到最后所有不包含选项的命令行参数都排到最后。

返回值

getopt()每次调用会逐次返回命令行传入的参数。
没有参数的最后的一次调用时,getopt()将返回-1
当解析到一个不在optstring里面的参数,或者一个必选值参数不带值时,返回?
当optstring是以:开头时,缺值参数的情况下会返回:,而不是? 。

范例

复制代码
 1 #include <unistd.h> 2 #include <stdlib.h> 3 #include <stdio.h> 4  5 int 6 main(int argc, char *argv[]) 7 { 8     int opt;    /*接收选项*/ 9     extern char* optarg;/*指向当前getopt()返回选项的参数*/10     extern int optopt;  /*当选项没有出现在optstring中,或者选项缺少必要的参数时,该选项存储在optopt中,getopt返回'?’*/11     extern int opterr;  /*用于控制getopt()是否打印出错信息*/12     extern int optind;  /*当前getopt()返回选项的下一个选项的索引(argv数组)*/13 14     opterr = 0; /*不要打印出错信息*/15 16     while ((opt = getopt(argc, argv, "a1b:c::")) != -1) {17         /* a和1为不带参数选项,b为必须带一个参数选项,c为可选参数选项(注意参数与-c直接不能分开) */18         /* 示例: getopt -a -b 100 -c12 */19         switch (opt) {20             case 'a':21             case '1':22                 printf("选项: %c\n",opt);23                 break;24             case 'b':25                 printf("选项: b,带的参数是 %s\n",optarg);26                 break;27             case 'c':28                 printf("选项: c,带的参数是 %s\n",optarg);29                 break;30             default: /* '?' */31                 if(optopt == 'c'){32                     printf("选项: c,没有带参数\n");33                     break;34                 }35                 fprintf(stderr, "用法: %s [-1a] [-c [argument]] [-b argument]\n",36                             argv[0]);37                 exit(EXIT_FAILURE); //无效的参数,退出程序38         }39     }40     printf("optind=%d\n",optind);41 42     //在命令行选项参数再也检查不到optstring中包含的选项时,43     //返回-1,同时optind储存第一个不包含选项的命令行参数。44     //getopt 中指的 选项是指以 `-`开头的45     if (optind >= argc) {46         fprintf(stderr, "选项索引超过了argv数组的长度\n");47         exit(EXIT_FAILURE);48     }49     //输出第一个不包含选项的参数50     printf("非选项参数 = %s\n", argv[optind]);51 52     //输出一下命令行参数,看看是否被改变53     for(opt = 0; opt < argc ; ++opt){54         printf("索引:%d\t\t命令行参数:%s\n",opt,argv[opt]);55     }56 57     exit(EXIT_SUCCESS); //成功退出58 }
复制代码

 

执行:

复制代码
fx@fx:~/code$ ./getopt -a -b 100 -c12选项: a选项: b,带的参数是 100选项: c,带的参数是 12optind=5选项索引超过了argv数组的长度fx@fx:~/code$ ./getopt -a 哈哈 -b 100 -c12 选项: a选项: b,带的参数是 100选项: c,带的参数是 12optind=5非选项参数 = 哈哈索引:0        命令行参数:./getopt索引:1        命令行参数:-a索引:2        命令行参数:-b索引:3        命令行参数:100索引:4        命令行参数:-c12索引:5        命令行参数:哈哈
复制代码

getopt_long示例代码

复制代码
 1 #include <unistd.h> 2 #include <stdlib.h> 3 #include <stdio.h> 4  5 extern int optind, opterr, optopt; 6  7 int 8 main(int argc, char **argv) 9 {10     int c;  /* 用于接收字符选项 */11     int digit_optind = 0;   /* 用于接收数字选项 */12 13     while (1) {14         /* */15         int this_option_optind = optind ? optind : 1;16         int option_index = 0;17         /* 长选项结构体数组 */18         static struct option long_options[] = {19             {"add",     required_argument, 0,  0 }, //需要一个参数20             {"append",  no_argument,       0,  0 }, //没有参数21             {"delete",  required_argument, 0,  0 },22             {"verbose", no_argument,       0,  0 },23             {"create",  required_argument, 0, 'c'}, //返回字符'c'24             {"file",    required_argument, 0,  0 },25             {0,         0,                 0,  0 }26         };27 /*28         struct option {29             const char *name; //选项名称30             int         has_arg;    //参数标志(no_argument/0没有参数;required_argument/1需要一个参数;optional_argument/2一个可选参数)31             int        *flag;   //指定如何返回一个较长的选项32             int         val;    //值33         };34 */35         /* 获取一个选项 */36         c = getopt_long(argc, argv, "abc:d:012",37                     long_options, &option_index);38 39         if (c == -1){   /* 无参数可获取了 */40           break;41         }42 43         switch (c) {    /* 获取参数解析 */44             case 0:45                 printf("选项是:%s", long_options[option_index].name);46                 if (optarg){    /*如果是带参数的选项 */47                   printf(" 参数是: %s", optarg);48                 }49                 printf("\n");50                 break;51 52             case '0':53             case '1':54             case '2':55                 if (digit_optind != 0 && digit_optind != this_option_optind)56                   printf("digits occur in two different argv-elements.\n");57                 digit_optind = this_option_optind;58                 printf("选项: %c\n", c);59                 break;60 61             case 'a':62                 printf("选项: a\n");63                 break;64 65             case 'b':66                 printf("选项: b\n");67                 break;68             case 'c':69                 printf("选项: c 带的值: '%s'\n", optarg);70                 break;71 72             case 'd':73                 printf("选项: d 带的值: '%s'\n", optarg);74                 break;75 76             case '?':77                 break;78 79             default:80                 printf("?? getopt 返回字符代码 0%o ??\n", c);81         }82     }83 84     if (optind < argc) {85         printf("非选项的命令行参数项: ");86         while (optind < argc)87           printf("%s ", argv[optind++]);88         printf("\n");89     }90     exit(EXIT_SUCCESS);91 }
0 0