Linux编程里getopt_long_only函数用法详解

时间:2021-03-31 04:41:51
在程序中难免需要使用命令行选项,可以选择自己解析命令行选项,但是有现成的,何必再造*。下面介绍使用getopt_long_only和getopt_long( 两者用法差不多 )解析命令行选项。

程序中主要使用:

短选项 长选项 是否需要参数
-n --username 是(用户名) 指定用户名
-d --debug 是否已测试

1、函数出处 

  1. #include <getopt.h> //getopt_long()头文件位置    
  2. int getopt_long (int ___argc, char *const *___argv,    
  3.             const char *__shortopts,    
  4.                 const struct option *__longopts, int *__longind);    
  5. int getopt_long_only (int ___argc, char *const *___argv,    
  6.                  const char *__shortopts,    
  7.                      const struct option *__longopts, int *__longind);   

2、参数介绍

  • argc argv :直接从main函数传递而来
  • shortopts:短选项字符串。如”n:v",这里需要指出的是,短选项字符串不需要‘-’(经测试需要),而且但选项需要传递参数时,在短选项后面加上“:”。
  • longopts:struct option 数组,用于存放长选项参数。
  • longind:用于返回长选项在longopts结构体数组中的索引值,用于调试。一般置为NULL
下面介绍struct option
  1. struct option    
  2. {    
  3.   const char *name;//长选项名    
  4.   int has_arg;//是否需要参数    
  5.   int *flag;    
  6.   int val;    
  7. };    
name:长选项名字 
has_arg:是否需要参数。值有三种情况

  1. # define no_argument        0    //不需要参数    
  2. # define required_argument  1    //必须指定参数    
  3. # define optional_argument  2    //参数可选    
flag和val相互依赖,主要分两种情况:(1)、flag为NULL,val值用于确定该长选项,所以需要为长选项指定唯一的val值。这里也为长选项和短选项建立了桥梁。            实际就是用于长选项和短选项的对应关系(2)、flag不为NULL,则将val值存放到flag所指向的存储空间,用于标识该长选项出现过。

3、返回值

  • 程序中使用短选项,则返回短选项字符(如‘n'),当需要参数是,则在返回之前将参数存入到optarg中。
  • 程序中使用长选项,返回值根据flag和val确定。当flag为NULL,则返回val值。所以根据val值做不同的处理,这也说明了val必须唯一。当val值等于短选项值,则可以使用短选项解析函数解析长选项;当flag不为NULL,则将val值存入flag所指向的存储空间,getopt_long返回0
  • 出现未定义的长选项或者短选项,getopt_long返回?
  • 解析完毕,getopt_long返回-1

4、实例

     理论要与实际相结合
实例一:
  1. #include <stdio.h>    
  2. #include <stdlib.h>    
  3. #include <getopt.h> //getopt_long()头文件位置    
  4.     
  5. int main(int argc, char** argv)    
  6. {    
  7.     const char *optstring="n:v";    
  8.     int c,deb,index;    
  9.     struct option opts[]={{"username",required_argument,NULL,'n'},    
  10.                           {"version",no_argument,NULL,'v'},    
  11.                           {"debug",no_argument,&deb,1},    
  12.                           {0,0,0,0}};    
  13.     while((c=getopt_long_only(argc,argv,optstring,opts,&index))!=-1)    
  14.     {    
  15.             
  16.         switch(c)    
  17.         {    
  18.         case 'n'://-n 或者 --username 指定用户名    
  19.             printf("username is %s\n",optarg);    
  20.             break;    
  21.         case 'v'://-v 或者--version,输出版本号    
  22.             printf("version is 0.0.1 \n");    
  23.             break;    
  24.         case 0://flag不为NULL    
  25.             printf("debug is %d\n",deb);    
  26.             break;    
  27.         case '?'://选项未定义    
  28.             printf("?\n");    
  29.             break;    
  30.         default:    
  31.             printf("c is %d\n",c);    
  32.             break;    
  33.         }    
  34.     }    
  35.     return 0;    
  36. }    

实例二:
  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3. #include <getopt.h> //getopt_long()头文件位置  
  4.   
  5. int main(int argc, char** argv)  
  6. {  
  7.     const char *optstring="n:v";  
  8.     int c,deb,index;  
  9.     struct option opts[]={{"username",required_argument,0,0},  
  10.                           {"n",required_argument,0,0},  
  11.                           {"version",no_argument,0,0},  
  12.                           {"v",no_argument,0,0},  
  13.                           {"debug",no_argument,0,0},  
  14.                           {"d",no_argument,0,0},  
  15.                           {"help",no_argument,0,0},  
  16.                           {"h",no_argument,0,0}};  
  17.     while((c=getopt_long_only(argc,argv,optstring,opts,&index))!=-1)  
  18.     {  
  19.         switch(index){  
  20.             //-n或者--username  
  21.             case 0:  
  22.             case 1:  
  23.                 printf("username:%s\n",optarg);  
  24.                 break;  
  25.             //-v或者--version  
  26.             case 2:  
  27.             case 3:  
  28.                 printf("version:1.0.0\n");  
  29.                 break;  
  30.             //-d or --debug  
  31.             case 4:  
  32.             case 5:  
  33.                 printf("debug:yes\n");  
  34.                 break;  
  35.             //-h or --help  
  36.             case 6:  
  37.             case 7:  
  38.                 printf("Help:?\n");  
  39.                 break;  
  40.             default:  
  41.                 printf("other:%d\n",index);  
  42.                 break;  
  43.         }  
  44.     }  
  45.     return 0;  
  46. }  

至于运行结果,我只给出用例,大家有兴趣可以试试。
实例1:./test -n boy
实例2:./test -n boy