原文地址:http://blog.csdn.net/cashey1991/article/details/7942809

getopt和getopt_long函数

 
平时在写程序时常常需要对命令行参数进行处理,当命令行参数个数较多时,如果按照顺序一个一个定义参数含义很容易造成混乱,而且如果程序只按顺序处理参数的话,一些“可选参数”的功能将很难实现。

在Linux中,我们可以使用getopt、getopt_long、getopt_long_only来对这个问题进行处理。

 #include <unistd.h>  

 int getopt(int argc, char * const argv[],
const char *optstring); extern char *optarg;
extern int optind, opterr, optopt; #include <getopt.h> int getopt_long(int argc, char * const argv[],
const char *optstring,
const struct option *longopts, int *longindex); int getopt_long_only(int argc, char * const argv[],
const char *optstring,
const struct option *longopts, int *longindex);

从最简单的getopt讲起,getopt函数的前两个参数,就是main函数的argc和argv,这两者直接传入即可,要考虑的就只剩下第三个参数。

optstring的格式举例说明比较方便,例如:

char *optstring = "abcd:";

上面这个optstring在传入之后,getopt函数将依次检查命令行是否指定了 -a, -b, -c及 -d(这需要多次调用getopt函数,直到其返回-1),当检查到上面某一个参数被指定时,函数会返回被指定的参数名称(即该字母)

最后一个参数d后面带有冒号,: 表示参数d是可以指定值的,如 -d 100 或 -d user。

optind表示的是下一个将被处理到的参数在argv中的下标值。

如果指定opterr = 0的话,在getopt、getopt_long、getopt_long_only遇到错误将不会输出错误信息到标准输出流。

 #include <unistd.h>
#include <stdlib.h>
#include <stdio.h> int main(int argc, char *argv[])
{
int opt;
char *optstring = "a:b:c:d"; while ((opt = getopt(argc, argv, optstring)) != -)
{
printf("opt = %c\n", opt);
printf("optarg = %s\n", optarg);
printf("optind = %d\n", optind);
printf("argv[optind - 1] = %s\n\n", argv[optind - ]);
} return ;
}

编译上述程序并运行,有如下结果:

 cashey@ubuntu:~/Desktop/getopt$ ./test_getopt -a  -b  -c admin -d
opt = a
optarg =
optind =
argv[optind - ] = opt = b
optarg =
optind =
argv[optind - ] = opt = c
optarg = admin
optind =
argv[optind - ] = admin opt = d
optarg = (null)
optind =
argv[optind - ] = -d

下面来讲getopt_long函数,getopt_long函数包含了getopt函数的功能,并且还可以指定“长参数”(或者说长选项),与getopt函数对比,getopt_long比其多了两个参数:

        int getopt(int argc, char * const argv[],
const char *optstring); int getopt_long(int argc, char * const argv[],
const char *optstring,
const struct option *longopts, int *longindex);

在这里,longopts指向的是一个由option结构体组成的数组,那个数组的每个元素,指明了一个“长参数”(即形如--name的参数)名称和性质:

           struct option {
const char *name;
int has_arg;
int *flag;
int val;
};

name  是参数的名称

has_arg 指明是否带参数值,其数值可选:

       no_argument (即 ) 表明这个长参数不带参数(即不带数值,如:--name)
required_argument (即 ) 表明这个长参数必须带参数(即必须带数值,如:--name Bob)
optional_argument(即2)表明这个长参数后面带的参数是可选的,(即--name和--name Bob均可)

flag   当这个指针为空的时候,函数直接将val的数值从getopt_long的返回值返回出去,当它非空时,val的值会被赋到flag指向的整型数中,而函数返回值为0

val    用于指定函数找到该选项时的返回值,或者当flag非空时指定flag指向的数据的值。

另一个参数longindex,如果longindex非空,它指向的变量将记录当前找到参数符合longopts里的第几个元素的描述,即是longopts的下标值。

 #include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <getopt.h> int
main(int argc, char **argv)
{
int opt;
int digit_optind = ;
int option_index = ;
char *optstring = "a:b:c:d";
static struct option long_options[] = {
{"reqarg", required_argument, NULL, 'r'},
{"noarg", no_argument, NULL, 'n'},
{"optarg", optional_argument, NULL, 'o'},
{, , , }
}; while ( (opt = getopt_long(argc, argv, optstring, long_options, &option_index)) != -)
{
printf("opt = %c\n", opt);
printf("optarg = %s\n", optarg);
printf("optind = %d\n", optind);
printf("argv[optind - 1] = %s\n", argv[optind - ]);
printf("option_index = %d\n", option_index);
} return ;
}

编译运行以上程序并运行,可以得到以下结果:

 cashey@ubuntu:~/Desktop/getopt$ ./test_getopt_long -a  --reqarg  --nonarg
opt = a
optarg =
optind =
argv[optind - ] =
option_index =
opt = r
optarg =
optind =
argv[optind - ] =
option_index =
./test_getopt_long: unrecognized option '--nonarg'
opt = ?
optarg = (null)
optind =
argv[optind - ] = --nonarg
option_index =

当所给的参数存在问题时,opt(即函数返回值是'?'),如:

 cashey@ubuntu:~/Desktop/getopt$ ./test_getopt_long -a
./test_getopt_long: option requires an argument -- 'a'
opt = ?
optarg = (null)
optind =
argv[optind - ] = -a
option_index =
cashey@ubuntu:~/Desktop/getopt$ ./test_getopt_long --reqarg
./test_getopt_long: option '--reqarg' requires an argument
opt = ?
optarg = (null)
optind =
argv[optind - ] = --reqarg

最后说说getopt_long_only函数,它与getopt_long函数使用相同的参数表,在功能上基本一致,只是getopt_long只将--name当作长参数,但getopt_long_only会将--name和-name两种选项都当作长参数来匹配。在getopt_long在遇到-name时,会拆解成-n -a -m -e到optstring中进行匹配,而getopt_long_only只在-name不能在longopts中匹配时才将其拆解成-n -a -m -e这样的参数到optstring中进行匹配。

 

最新文章

  1. [LeetCode] Merge k Sorted Lists 合并k个有序链表
  2. 【BZOJ1006】【HNOI2008】神奇的国度(弦图染色)
  3. java项目中读取src目录下的文件
  4. .Net 代码安全保护产品DNGuard HVM使用
  5. 1.3 LINQ查询
  6. 小学生之Hibernate插入数据修改数据使用数据库默认值的实现
  7. svn本地目录结构for window
  8. java ee Servlet 开发框架分享
  9. java数据库编程之初始Mysql
  10. ATL右键文件菜单
  11. 【druid 】数据库连接池
  12. python numpy 三行代码打乱训练数据
  13. IDEA发布应用时发布到lib下面的包不全
  14. UVa 10003 切木棍(区间DP+最优矩阵链乘)
  15. IocPerformance 常见IOC 功能、性能比较
  16. jq如何判断是否存在某个指定的style样式
  17. HAWQ + MADlib 玩转数据挖掘之(一)——安装
  18. Beta阶段个人总结
  19. linux后台运行之screen和nohup
  20. C语言第十一回合:预处理命令的集中营

热门文章

  1. Android studio安装配置常见问题及其解决方案
  2. MFC中onmouseover与onmousemove的区别
  3. JAVA中如何将一个json形式的字符串转为json对象或对象列表
  4. java--线程认识与实例记录 NO.1
  5. Python 各种测试框架简介(三):nose
  6. 详解Nginx + Tomcat 反向代理 如何在高效的在一台服务器部署多个站点
  7. 解决 IllegalArgumentException: Could not resolve placeholder in string value &quot;${XXXXXX}&quot;
  8. Linux下使用Supervisor来管理维护程序-详解
  9. 【HDU 5647】DZY Loves Connecting(树DP)
  10. poj 1664 放苹果(递推)