命令行参数解析之getopt

2023-12-06 06:08
文章标签 参数 解析 命令行 getopt

本文主要是介绍命令行参数解析之getopt,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文作者: 阿宝 | 彩色世界
本文链接: https://blog.ibaoger.com/2017/08/08/getopt-long/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 3.0 许可协议。转载请注明出处!

简介

getopt作为Linux下经典的命令行解析工具,得到广泛的使用,下面简单的作一下介绍。

定义

getopt提供了4个变量、1个结构体、3个外部接口、1个内部接口:

char *optarg;
int optind;
int opterr;
int optopt;struct option
{
#if defined (__STDC__) && __STDC__const char *name;
#elsechar *name;
#endifint has_arg;int *flag;int val;
};int getopt (int argc, char *const *argv, const char *optstring)int getopt_long (int argc,  char *const *argv,  const char *options,const struct option *long_options, int *opt_index)int getopt_long_only (int argc, char *const *argv, const char *options,const struct option *long_options, int *opt_index)int _getopt_internal (int argc, char *const *argv, const char *optstring,const struct option *longopts, int *longind, int long_only)

optarg

optarg用于从getopt到调用者的交互。
当 getopt 找到带参数的选项时,optarg 被赋值为这个参数;当枚举变量 ordering 等于 RETURN_IN_ORDER 时,没有选项的参数赋值给 optarg。

optind

ARGV中将要被处理的下一个元素的下标。用于和调用者交互,并连续调用getopt。
在进入getopt时,零表示这是第一个调用; 初始化。当getopt返回-1时,表示这是第一个没有选项的参数,需要调用者自己做处理。正常情况下,optind表示ARGV中已经处理的元素个数。

opterr

当getopt遇到无法识别的选项时,调用者通过给opterr赋值零来阻止打印错误消息。

optopt

optopt表示无法识别的选项字符。

struct option

描述应用程序中使用到的长名称选项。
getopt_long或getopt_long_only方法中的LONG_OPTIONS参数,是一个option结构体的容器,并以名称为零的元素结尾。

has_arg
no_argument(0) 代表此选项没有参数;
required_argument(1) 代表此选项需要参数;
optional_argument(2) 代表此选项的参数可有可无;

flag
当flag不等于零时,如果找到选项,那么选项的参数被赋值为val变量;否则不作任何修改;

在使用长名称选项时,除了将int设置为编译常量之外,还需要设置一个来自optarg的值,设置flag为零,设置val为非零(等效的单字母选项,如果有的话)。对于flag字段为零的长选项,getopt返回val字段的值。

getopt

短名称选项的解析。

getopt_long

兼容长名称和短名称选项的解析。

getopt_long_only

长名称选项的解析。

_getopt_internal

长短名称选项的内部解析实现。从给定的OPTSTRING字符串中,搜索(长度为ARGC)ARGV的元素中相匹配的字符选项。

如果ARGV元素以 - 符号开始,并且不等于 - 或 – 那么这个元素是选项,这个元素中的字符是选项的名称。当调用循环getopt时,将成功返回每个元素中的字符。

当getopt找到选项时,将返回元素中的字符,并更新optind和nextchar变量,以便下次搜索时使用。

当查找完成后,getopt返回-1。此时optind被赋值为ARGV中第一个非选项的下标。

OPTSTRING是一个包含合法选项字符的字符串。如果ARGV中的元素不在OPTSTRING中时,getopt输出错误住处,并返回?。当设置opterr为零时,不输出错误,仍返回?。如果OPTSTRING以 - 或 + 符号开始,那么你需要自己处理非选项的ARGV元素。详情参阅RETURN_IN_ORDER和REQUIRE_ORDER的说明。

长名称选项以 – 符号开始,名称可以使用全称或缩写,并保证名称的唯一性。长选项的参数,可以放在长选项的下一个元素,也可以通过 = 紧跟在长选项的后面。当getopt'发现一个长命名选项时,如果该选项的flag’字段为非零,则返回零,如果’flag’字段为零,则返回该选项的’val’字段的值。

ARGV中的元素并不是真正的常量。我们当作常量来使用,以保证与其他系统的兼容性。

LONGOPTS是一个option结构体的容器,并以名称为零的元素结尾。
LONGIND代表长名称选项在LONGOPT中的下标。仅在长名称选项匹配成功时有效。
LONG_ONLY为非零时,- 符号和 – 符号一样代表长名称的起始。

示例

int main(int argc, char **argv)
{int c;int digit_optind = 0;while (1){int this_option_optind = optind ? optind : 1;int option_index = 0;static struct option long_options[] ={{"add", 1, 0, 0},{"append", 0, 0, 0},{"delete", 1, 0, 0},{"verbose", 0, 0, 0},{"create", 0, 0, 0},{"file", 1, 0, 0},{0, 0, 0, 0}};c = getopt_long(argc, argv, "abc:d:0123456789",long_options, &option_index);if (c == -1)break;switch (c){case 0:printf("option %s", long_options[option_index].name);if (optarg)printf(" with arg %s", optarg);printf("\n");break;case '0':case '1':case '2':case '3':case '4':case '5':case '6':case '7':case '8':case '9':if (digit_optind != 0 && digit_optind != this_option_optind)printf("digits occur in two different argv-elements.\n");digit_optind = this_option_optind;printf("option %c\n", c);break;case 'a':printf("option a\n");break;case 'b':printf("option b\n");break;case 'c':printf("option c with value `%s'\n", optarg);break;case 'd':printf("option d with value `%s'\n", optarg);break;case '?':break;default:printf("?? getopt returned character code 0%o ??\n", c);}}if (optind < argc){printf("non-option ARGV-elements: ");while (optind < argc)printf("%s ", argv[optind++]);printf("\n");}exit(0);
}

这篇关于命令行参数解析之getopt的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/460717

相关文章

全面解析Golang 中的 Gorilla CORS 中间件正确用法

《全面解析Golang中的GorillaCORS中间件正确用法》Golang中使用gorilla/mux路由器配合rs/cors中间件库可以优雅地解决这个问题,然而,很多人刚开始使用时会遇到配... 目录如何让 golang 中的 Gorilla CORS 中间件正确工作一、基础依赖二、错误用法(很多人一开

Mysql中设计数据表的过程解析

《Mysql中设计数据表的过程解析》数据库约束通过NOTNULL、UNIQUE、DEFAULT、主键和外键等规则保障数据完整性,自动校验数据,减少人工错误,提升数据一致性和业务逻辑严谨性,本文介绍My... 目录1.引言2.NOT NULL——制定某列不可以存储NULL值2.UNIQUE——保证某一列的每一

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

MySQL CTE (Common Table Expressions)示例全解析

《MySQLCTE(CommonTableExpressions)示例全解析》MySQL8.0引入CTE,支持递归查询,可创建临时命名结果集,提升复杂查询的可读性与维护性,适用于层次结构数据处... 目录基本语法CTE 主要特点非递归 CTE简单 CTE 示例多 CTE 示例递归 CTE基本递归 CTE 结

Spring Boot 3.x 中 WebClient 示例详解析

《SpringBoot3.x中WebClient示例详解析》SpringBoot3.x中WebClient是响应式HTTP客户端,替代RestTemplate,支持异步非阻塞请求,涵盖GET... 目录Spring Boot 3.x 中 WebClient 全面详解及示例1. WebClient 简介2.

在MySQL中实现冷热数据分离的方法及使用场景底层原理解析

《在MySQL中实现冷热数据分离的方法及使用场景底层原理解析》MySQL冷热数据分离通过分表/分区策略、数据归档和索引优化,将频繁访问的热数据与冷数据分开存储,提升查询效率并降低存储成本,适用于高并发... 目录实现冷热数据分离1. 分表策略2. 使用分区表3. 数据归档与迁移在mysql中实现冷热数据分

C#解析JSON数据全攻略指南

《C#解析JSON数据全攻略指南》这篇文章主要为大家详细介绍了使用C#解析JSON数据全攻略指南,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、为什么jsON是C#开发必修课?二、四步搞定网络JSON数据1. 获取数据 - HttpClient最佳实践2. 动态解析 - 快速

Spring Boot3.0新特性全面解析与应用实战

《SpringBoot3.0新特性全面解析与应用实战》SpringBoot3.0作为Spring生态系统的一个重要里程碑,带来了众多令人兴奋的新特性和改进,本文将深入解析SpringBoot3.0的... 目录核心变化概览Java版本要求提升迁移至Jakarta EE重要新特性详解1. Native Ima

spring中的@MapperScan注解属性解析

《spring中的@MapperScan注解属性解析》@MapperScan是Spring集成MyBatis时自动扫描Mapper接口的注解,简化配置并支持多数据源,通过属性控制扫描路径和过滤条件,利... 目录一、核心功能与作用二、注解属性解析三、底层实现原理四、使用场景与最佳实践五、注意事项与常见问题六

nginx -t、nginx -s stop 和 nginx -s reload 命令的详细解析(结合应用场景)

《nginx-t、nginx-sstop和nginx-sreload命令的详细解析(结合应用场景)》本文解析Nginx的-t、-sstop、-sreload命令,分别用于配置语法检... 以下是关于 nginx -t、nginx -s stop 和 nginx -s reload 命令的详细解析,结合实际应