C语言处理参数的 getopt() 函数

本文详细介绍了C语言中的getopt函数,包括其参数传递机制、处理选项和参数的方式、扫描模式以及错误处理策略。通过实例演示了如何使用getopt处理命令行选项,适合理解和实践命令行参数解析。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

C语言程序主要通过 main 函数的参数来传递命令行参数:

// 默认传递命令行参数 
int main(int argc ,char * argv[]) 
{ 
    ... 
}

其中 argc 表示参数个数(包含程序本身),argv是保存所有这些参数的二级数组。

我们可以直接操作这两个变量来处理参数,但是C标准库提供了一个更丰富的处理参数的方法,就是 getopt()。

getopt 介绍

函数声明如下:

#include <unistd.h>

int getopt(int argc, char * const argv[], const char *optstring);

getopt() 函数用于解析命令行参数。它的 argc 和 argv 参数通常直接从 main() 的参数直接传递而来。

optstring 是可以处理选项字母组成的字符串。该字串里的每个字符对应于一个以 ‘-’ 开头的选项。如果该字串里的任一字符后面有冒号,那么这个选项就要求有参数(如“hd:”对应于 '-h' 和 '-d', 其中 '-d' 后需接参数)。而如果选项后面接两个冒号,则说明这个选项后的参数是可选的,即可带参数也可不带参数。

全局变量

当给定 getopt() 命令参数的数量 (argc)、指向这些参数的数组 (argv) 和选项字串 (optstring) 后,getopt() 将返回第一个选项,并设置一些全局变量。

extern char *optarg;
extern int optind, opterr, optopt;
  • char *optarg : 指向当前选项的参数字串(如果有)
  • int optind : 变量 optind 是 argv 中要处理的下一个元素的索引。系统将此值初始化为 1。该值随着 getopt() 的调用而变化,调用者可以将其重置为 1 以重新开始扫描相同的 argv,或者扫描新的参数向量。
  • int opterr : 错误信息打印标志,这个变量非零时,getopt() 函数会为"无效选项"和"缺少参数选项"两种情况输出其错误信息.
  • int optopt : 当发现无效选项字符之时,optopt包含了所发现的无效选项字符

处理机制

getopt() 调用时会根据 optind 和 optstring 遍历选项和参数,如果找到一个选项字符,则返回该字符,同时更新全局变量 optind,optarg 以及静态变量 nextchar,以便下一次调用 getopt() 时可以继续扫描。

如果 getopt() 被重复调用,它会依次返回每个选项字符。如果所有可识别的选项都被扫描过,将返回 -1,此时 optind 是第一个不是选项的 argv 元素的索引。

扫描模式

默认情况下,getopt() 会在扫描时置换 argv 的内容,以便扫描完成后所有的非选项都处于末尾。但还提供了另外两种扫描模式。

1. 一旦遇到非选项参数就立即停止,通过 optstring 的一个字符设置为 '+' 来选择这种模式;

2. 每个非选项 argv 元素都被处理为好像它是具有字符代码 1 的选项的参数一样,此时碰到那些没有选项对应的参数时不会报错也不会退出,而是认为它是选项“1”的参数。通过 optstring 的第一个字符设置为 ‘-’ 来选择这种模式。

注:无论处于那种扫描模式,碰到参数 '--' 都会强制结束选项扫描。

getopt出错处理机制

在处理选项列表时,getopt() 可以检测到两种错误:

1. 未在 optstring 中定义的选项字符;

2. 缺少选项参数;

默认情况下, getopt() 会打印一条错误消息,将错误的选项字符放在 optopt 中,并返回 '?' 作为函数结果。但如果我们想要其他的处理方式,可以有以下做法:

1. 在调用 getopt() 前设置 opterr 为零,此时 getopt() 不会输出错误信息,我们可以通过返回值是否为 '?' 来进行不同的处理。

2. 将 optstring 的第一个字符设置为 ‘+’ 或者 '-',此时 getopt() 同样不会输出错误信息,同时可以根据返回值是 ‘?’ 还是 ':' 来做不同的处理。 ‘?’ 表示未定义选项,':' 表示缺少选项参数

示例

以下简单的示例程序使用 getopt() 处理两个程序选项: -n,没有关联值;和 -t val,它需要一个关联的值。

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>

int main(int argc, char *argv[])
{
   int flags, opt;
   int nsecs, tfnd;

   nsecs = 0;
   tfnd = 0;
   flags = 0;
   while ((opt = getopt(argc, argv, "nt:")) != -1) {
       switch (opt) {
       case 'n':
           flags = 1;
           break;
       case 't':
           nsecs = atoi(optarg);
           tfnd = 1;
           break;
       default: /* '?' */
           fprintf(stderr, "Usage: %s [-t nsecs] [-n] name\n",
                   argv[0]);
           exit(EXIT_FAILURE);
       }
   }

   printf("flags=%d; tfnd=%d; nsecs=%d; optind=%d\n",
           flags, tfnd, nsecs, optind);

   if (optind >= argc) {
       fprintf(stderr, "Expected argument after options\n");
       exit(EXIT_FAILURE);
   }

   printf("name argument = %s\n", argv[optind]);

   exit(EXIT_SUCCESS);
}

参考链接

getopt函数_sdoyuxuan的博客-CSDN博客_getopt函数

man entry of getopt

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

翔底

您的鼓励将是我创作最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值