【C】命令行参数解析——getopt、getopt_long及getopt_long_only
前言
在linux下学习开源代码Webbench,遇到get_long等函数的用法,一时有点懵,故想深入了解这类命令行解析函数,并记此博文。
1、getopt
getopt主要用来处理短命令行选项,例如 ./test -v 中的 -v 就是一个短选项。使用该函数需引入头文件<unistd.h>,下面是该函数定义:
int getopt (int ___argc, char *const *___argv, const char *__shortopts);
其中___argc和___argv是main函数中传递的参数个数和内容,__shortopts用来指定可以处理哪些选项,以下为__shortopts的示例:
"a:bc"
该示例表明该程序可以接受3个选项:-a -b -c ,其中a后面的 :表示该选项后面跟一个参数,即如 -a text 的形式,选项后面跟的参数会被保存到optarg变量中。下面是一个使用的示例:
#include <stdio.h>
#include <unistd.h>
void use_getopt(int argc,char **argv){
int ch;
){
switch(ch){
case 'a':
printf("option a: %s\n",optarg);
break;
case 'b':
printf("option b: \n");
break;
case '?':
printf("unknown option \n");
break;
default:
printf("default");
}
}
}
int main(int argc,char **argv){
use_getopt(argc,argv);
;
}
执行 ./test -a aa -b -c 输出结果如下:
option a: aa option b: ./test: invalid option -- 'c' unknown option
2、getopt_long
getopt_long支持长选项的命令行解析,所谓长选项即诸如 --help 的形式,使用该函数需要引入头文件<getopt.h>,下面是getopt_long和getopt_long_only的函数原型:
#include <getopt.h>
int getopt_long (int ___argc,
char *const *___argv,
const char *__shortopts,
const struct option *__longopts,
int *__longind);
int getopt_long_only (int ___argc,
char *const *___argv,
const char *__shortopts,
const struct option *__longopts,
int *__longind);
其中___argc,___argv, __shortopts 和 getopt 中的含义一样,下面解释 __longopts 和 __longind.
__longopts指向一个 struct option 的数组,下面是option的定义:
struct option
{
const char *name;
int has_arg;
int *flag;
int val;
};
- name — 长选项的名称,例如 help
- has_arg — 是否带参数,0不带参数,1必须带参数,2参数可选
- flag — 指定长选项如何返回结果,如果flag为NULL, getopt_long()会返回val. 如果flag不为NULL,getopt_long会返回0,并且将val的值存储到flag中
- val — 将要被getopt_long返回或者存储到flag指向的变量中的值
下面是一个使用示例:
struct option opts[] = {
{, NULL, 'v'},
{, NULL, 'n'},
{, NULL, 'h'}
};
以上{"version", 0, NULL, 'v'}, version 即为长选项的名称, 即按如下形式 --version,,0 表示该选项后面不带参数,NULL 表示直接将v返回(字符v在ascii码中对应的数值), 即在使用getopt_long遍历到该条选项时,getopt_long 返回值为字符v对应的ascii码值。
__longind表示长选项在__longopts中的位置,例如上面的例子中,version对应的__longind为0,name对应的__longind为1,help对应的__longind为2,该项主要用于调试,一般设为NULL即可。
下面是一个使用示例:
void use_getopt_long(int argc,char **argv){
const char *optstring = "vn:h";
int c;
struct option opts[] = {
{, NULL, 'v'},
{, NULL, 'n'},
{, NULL, 'h'}
};
){
switch(c){
case 'n':
printf("name is %s\n",optarg);
break;
case 'v':
printf("verson is 0.0.1\n");
break;
case 'h':
printf("this is help\n");
break;
case '?':
printf("unknown option\n");
break;
:
printf("the return val is 0\n");
break;
default:
printf("default");
}
}
}
运行程序 ./test --name evenleo --version --help --haha ,结果如下:
name is evenleo verson is 0.0.1 this is help ./test: unrecognized option '--haha' unknown option
当然也可以使用短选项 ./test -n evenleo -v -h,结果如下:
name is evenleo verson is 0.0.1 this is help
将上面的程序进行修改,将struct option中的flag和__longind设为具体值
void use_getopt_long2(int argc, char *argv[]) {
const char *optstring = "vn:h";
int c;
, f_n = -, f_h = -, opt_index = -;
struct option opts[] = {
{, &f_v, 'v'},
{, &f_n, 'n'},
{, &f_h, 'h'}
};
) {
switch(c) {
case 'n':
printf("name is %s\n", optarg);
break;
case 'v':
printf("version is 0.0.1\n");
break;
case 'h':
printf("this is help\n");
break;
case '?':
printf("unknown option\n");
break;
:
printf("f_v is %d \n", f_v);
printf("f_n is %d \n", f_n);
printf("f_h is %d \n", f_h);
break;
default:
printf("------\n");
}
printf("opt_index is %d\n\n", opt_index);
}
}
运行 ./test -name evenleo -version -help,结果如下:
f_v is -1 f_n is 110 f_h is -1 opt_index is 1 f_v is 118 f_n is 110 f_h is -1 opt_index is 0 f_v is 118 f_n is 110 f_h is 104 opt_index is 2
以上结果可以看出,当给flag指定具体的指针后,getopt_long会返回0,并且val的值赋给了flag指向的变量,下面我们用短选项执行一下程序,运行 ./test -n evenleo -v -h ,结果如下:
name is evenleo opt_index is -1 version is 0.0.1 opt_index is -1 this is help opt_index is -1
可以看出使用短选项时getopt_long相当于getopt,flag和__longind不起作用。
3、getopt_long和getopt_long_only区别
下面解释一下 getopt_long 和 getopt_long_only的区别,用use_getopt_long函数 ,运行 ./test -name evenleo -verson -help,结果如下:
name is ame
verson is 0.0.1
./test: invalid option -- 'e'
unknown option
./test: invalid option -- 'r'
unknown option
./test: invalid option -- 's'
unknown option
./test: invalid option -- 'i'
unknown option
./test: invalid option -- 'o'
unknown option
name is -help
可以看出使用短选项标识符 - 指向长选项时,程序还是会按短选项来处理,即一个字符一个字符的解析。下面将函数use_getopt_long做一些更改,即将getopt_long改为getopt_long_only,如下:
void use_getopt_long3(int argc,char **argv){
const char *optstring = "vn:h";
int c;
struct option opts[] = {
{, NULL, 'v'},
{, NULL, 'n'},
{, NULL, 'h'}
};
){
switch(c){
case 'n':
printf("name is %s\n",optarg);
break;
case 'v':
printf("verson is 0.0.1\n");
break;
case 'h':
printf("this is help\n");
break;
case '?':
printf("unknown option\n");
break;
:
printf("the return val is 0\n");
break;
default:
printf("default");
}
}
}
再运行 ./test -name evenleo -version -help,结果如下:
name is evenleo verson is 0.0.1 this is help
即使用 getopt_long_only 时, - 和 -- 都可以作用于长选项, 而使用 getopt_long 时,只有 -- 可以作用于长选项.
文章大部分内容参考http://blog.zhangjikai.com/2016/03/05/%E3%80%90C%E3%80%91%E8%A7%A3%E6%9E%90%E5%91%BD%E4%BB%A4%E8%A1%8C%E5%8F%82%E6%95%B0--getopt%E5%92%8Cgetopt_long/
【C】命令行参数解析——getopt、getopt_long及getopt_long_only的更多相关文章
- Python命令行参数解析模块getopt使用实例
Python命令行参数解析模块getopt使用实例 这篇文章主要介绍了Python命令行参数解析模块getopt使用实例,本文讲解了使用语法格式.短选项参数实例.长选项参数实例等内容,需要的朋友可以参 ...
- python命令行参数解析模块argparse和docopt
http://blog.csdn.net/pipisorry/article/details/53046471 还有其他两个模块实现这一功能,getopt(等同于C语言中的getopt())和弃用的o ...
- shell 命令行参数(getopt和getopts)
getopt 命令 使用getopt命令,可以解析任何命令行选项和参数,但是用法比较复杂.getopt的命令用法如下: $ getopt --help 用法: getopt optstring par ...
- [转]Python 命令行参数和getopt模块详解
FROM : http://www.tuicool.com/articles/jaqQvq 有时候我们需要写一些脚本处理一些任务,这时候往往需要提供一些命令行参数,根据不同参数进行不同的处理,在Pyt ...
- Python 命令行参数和getopt模块详解
有时候我们需要写一些脚本处理一些任务,这时候往往需要提供一些命令行参数,根据不同参数进行不同的处理,在Python里,命令行的参数和C语言很类似(因为标准Python是用C语言实现的).在C语言里,m ...
- gflags命令行参数解析
gflags库是google开源的命令行参数解析工具. 安装 官方没有提供二进制库,但是Debian/Ubuntu平台本身提供了二进制库,可以直接git clone https://github.co ...
- [Go] 命令行参数解析包(flag 包)使用详解
Go 的 flag 包可以解析命令行的参数. 一.命令行语法 命令行语法主要有以下几种形式: cmd -flag // 只支持bool类型 cmd -flag=xxx cmd -flag ...
- $命令行参数解析模块argparse的用法
argparse是python内置的命令行参数解析模块,可以用来为程序配置功能丰富的命令行参数,方便使用,本文总结一下其基本用法. 测试脚本 把以下脚本存在argtest.py文件中: # codin ...
- Google开源命令行参数解析库gflags
Google开源命令行参数解析库gflags http://blog.csdn.net/lming_08/article/details/25072899 CMDLINE的解析 http://blog ...
随机推荐
- vue路由跳转push,replace,go
this.$router.replace({ path: "/subpagest" });//不会向 history 添加新记录,而是跟它的方法名一样 —— 替换掉当前的 his ...
- java常用数据类型转换
在Java开发过程中经常会涉及到数据类型的转换问题,比如数字型转字符型,字符型转日期型,字符串转数组等等,以及其他类型的强制转换等.经常出现,所以有必要总结一下. 1.如何将字串 String 转换成 ...
- springboot错误1 Failed to execute goal org.springframework.boot:spring-boot-maven-plugin
关于Springboot打包错误的问题 | Failed to execute goal org.springframework.boot:spring-boot-maven-plugin https ...
- JVM(九),垃圾回收回收算法
九.垃圾回收回收算法 1.标记-清除(Mark and Sweep) 缺点是内存空间碎片化太严重 2.复制算法(Copying) (1)复制算法介绍 (2)复制算法优势 3.标记-整理算法(Compa ...
- vue中全局组件与局部组件的注册,以及动态绑定props值
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Java线程之synchronized
翻译:https://www.journaldev.com/1061/thread-safety-in-java 我们知道线程安全在Java中是一个非常重要的主题,当多个线程操作共享数据时,如果没有任 ...
- 【知识库】-数据库_MySQL 的七种 join
掘金作者:haifeisi 文章出处: MySQL 的七种 join Learn [已经过测试校验] 一.内连接 二.左外连接 三.右外连接 四.左连接 五.右连接 六.全连接 七.两张表中都没有出现 ...
- 记一次期待已久的渗透 从phpcms到thinkphp
0X01 前言 这是刚刚开始学习渗透的一个目标吧 这个站从刚开始学的那一天起,就想把他日下来. 可能是自己的信息收集能力太差了吧,导致一直无从下手 没有进展.这是需要慢慢积累的过程.还需努力学习. 0 ...
- [集训队作业2018]蜀道难——TopTree+贪心+树链剖分+链分治+树形DP
题目链接: [集训队作业2018]蜀道难 题目大意:给出一棵$n$个节点的树,要求给每个点赋一个$1\sim n$之内的权值使所有点的权值是$1\sim n$的一个排列,定义一条边的权值为两端点权值差 ...
- flask 第四篇 模板语言jinja2
是时候开始写个前端了,Flask中默认的模板语言是Jinja2 现在我们来一步一步的学习一下 Jinja2 捎带手把 render_template 中留下的疑问解决一下 首先我们要在后端定义几个字符 ...