Redian新闻
>
在 C 语言中使用 getopt 解析命令行短选项 | Linux 中国

在 C 语言中使用 getopt 解析命令行短选项 | Linux 中国

科技
 
导读:通过使用命令行让用户告诉程序要什么,可以让程序更加灵活。                             
本文字数:5492,阅读时长大约:7分钟

在已经知道要处理什么文件和对文件进行哪些操作的情况下,编写处理文件的 C 语言程序就很容易了。如果将文件名“硬编码”在程序中,或者你的程序只以一种方式来处理文件,那么你的程序总是知道要做什么。

但是如果程序每次运行时能够对用户的输入做出反应,可以使程序更灵活。让用户告诉程序要处理什么文件,或者以不同的方式完成任务,要实现这样的功能就需要读取命令行参数。

读取命令行

一个 C 语言程序可以用如下声明开头:

  1. int main()

这是启动 C 程序最简单的形式。但如果在圆括号中加入标准参数,你的程序就可以从命令行中读取选项了:

  1. int main(int argc, char **argv)

argc 表示命令行中的参数个数。它总是一个至少为 1 的数。

argv 是一个二级指针,它指向一个字符串数组。这个数组中保存的是从命令行接收的各个参数。数组的第一个元素 *argv[0] 是程序的名称。**argv 数组的其它元素包含剩下的命令行参数。

下面我将写一个简单的示例程序,它能够回显通过命令行参数传递给它的选项。它跟 Linux 的 echo 命令类似,只不过我们的程序会打印出程序名。同时它还会调用 puts 函数将命令行选项按行打印输出。

  1. #include <stdio.h>
  2. int
  3. main(int argc, char **argv)
  4. {
  5. int i;
  6. printf("argc=%d\n", argc); /* debugging */
  7. for (i = 0; i < argc; i++) {
  8. puts(argv[i]);
  9. }
  10. return 0;
  11. }

编译此程序,并在运行时提供一些命令行参数,你会看到传入的命令行参数被逐行打印出来:

  1. $ ./echo this program can read the command line
  2. argc=8
  3. ./echo
  4. this
  5. program
  6. can
  7. read
  8. the
  9. command
  10. line

这个命令行将程序的 argc 置为 8,**argv 数组包含 8 个元素:程序名以及用户输入的 7 个单词。由于 C 语言中数组下标从 0 开始,所以这些元素的标号分别是 0 到 7。这也是在 for 循环中处理命令行参数时能够用 i < argc 作为比较条件的原因。

你也可以用这个方式实现自己的 cat 或 cp 命令。cat 命令的基本功能是显示一个或几个文件的内容。下面是一个简化版的cat 命令,它从命令行获取文件名:

  1. #include <stdio.h>
  2. void
  3. copyfile(FILE *in, FILE *out)
  4. {
  5. int ch;
  6. while ((ch = fgetc(in)) != EOF) {
  7. fputc(ch, out);
  8. }
  9. }
  10. int
  11. main(int argc, char **argv)
  12. {
  13. int i;
  14. FILE *fileptr;
  15. for (i = 1; i < argc; i++) {
  16. fileptr = fopen(argv[i], "r");
  17. if (fileptr != NULL) {
  18. copyfile(fileptr, stdout);
  19. fclose(fileptr);
  20. }
  21. }
  22. return 0;
  23. }

这个简化版的 cat 命令从命令行读取文件名列表,然后将各个文件的内容逐字符地显示到标准输出上。假定我有一个叫做 hello.txt 的文件,其中包含数行文本内容。我能用自己实现的 cat 命令将它的内容显示出来:

  1. $ ./cat hello.txt
  2. Hi there!
  3. This is a sample text file.

以这个简单程序为出发点,你也可以实现自己版本的其它 Linux 命令。比如 cp 命令,它从命令行读取两个文件名:要读取的文件和要写入的文件。

读取命令行选项

通过命令行读取文件名和其它文本固然很棒,但是如果想要程序根据用户给出的选项改变行为呢?比如 Linux 的 cat 命令就支持以下命令行选项:

◈ -b 显示非空行的行号
◈ -E 在行尾显示 $
◈ -n 显示行号
◈ -s 合并显示空行
◈ -T 将制表符显示为 ^I
◈ -v 用 ^x 和 M-x 方式显示非打印字符,换行符和制表符除外

这些以一个连字符开头的单字母的选项叫做短选项。通常短选项是分开使用的,就像这样 cat -E -n。但是也可以将多个短选项合并,比如 cat -En

值得庆幸的是,所有 Linux 和 Unix 系统都包含 getopt 库。它提供了一种简单的方式来读取命令行参数。getopt 定义在头文件 unistd.h 中。你可以在程序中使用 getopt 来读取命令行短选项。

与其它 Unix 系统不同的是,Linux 上的 getopt 总是保证短选项出现在命令行参数的最前面。比如,用户输入的是 cat -E file -n-E 在最前面,-n 在文件名之后。如果使用 Linux 的 getopt 来处理,程序会认为用户输入的是 cat -E -n file。这样做可以使处理过程更顺畅,因为 getopt 可以解析完所有短选项,剩下的文件名列表可以通过 **argv 来统一处理。

你可以这样使用 getopt:

  1. #include <unistd.h>
  2. int getopt(int argc, char **argv, char *optstring);

optstring 是由所有合法的选项字符组成的字符串。比如你的程序允许的选项是 -E 和 -n, 那么 optstring 的值就是 "En"

通常通过在循环中调用 getopt 来解析命令行选项。每次调用时 getopt 会返回找到的下一个短选项,如果遇到无法识别的选项则返回 '?'。当没有更多短选项时它返回 -1,并且设置全局变量 optind 的值指向 **argv 中所有段选项之后的第一个元素。

下面看一个简单的例子。这个演示程序没有实现 cat 命令的所有选项,但它只是能够解析命令行。每当发现一个合法的命令行选项,它就打印出相应的提示消息。在你自己的程序中,你可能会根据这些命令行选项执行变量赋值等者其它操作。

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. int
  4. main(int argc, char **argv)
  5. {
  6. int i;
  7. int option;
  8. /* parse short options */
  9. while ((option = getopt(argc, argv, "bEnsTv")) != -1) {
  10. switch (option) {
  11. case 'b':
  12. puts("Put line numbers next to non-blank lines");
  13. break;
  14. case 'E':
  15. puts("Show the ends of lines as $");
  16. break;
  17. case 'n':
  18. puts("Put line numbers next to all lines");
  19. break;
  20. case 's':
  21. puts("Suppress printing repeated blank lines");
  22. break;
  23. case 'T':
  24. puts("Show tabs as ^I");
  25. break;
  26. case 'v':
  27. puts("Verbose");
  28. break;
  29. default: /* '?' */
  30. puts("What's that??");
  31. }
  32. }
  33. /* print the rest of the command line */
  34. puts("------------------------------");
  35. for (i = optind; i < argc; i++) {
  36. puts(argv[i]);
  37. }
  38. return 0;
  39. }

假如你把程序编译为 args,你可以通过尝试不同的命令行参数组合,来了解程序是怎么解析短选项,以及是怎么将其它的命令行参数留下来的。最简单的例子是将所有的选项都放在最前面,就像这样:

  1. $ ./args -b -T file1 file2
  2. Put line numbers next to non-blank lines
  3. Show tabs as ^I
  4. ------------------------------
  5. file1
  6. file2

现在试试将两个短选项合并使用的效果:

  1. $ ./args -bT file1 file2
  2. Put line numbers next to non-blank lines
  3. Show tabs as ^I
  4. ------------------------------
  5. file1
  6. file2

如果有必要的话,getopt可以对命令行参数进行重排:

  1. $ ./args -E file1 file2 -T
  2. Show the ends of lines as $
  3. Show tabs as ^I
  4. ------------------------------
  5. file1
  6. file2

如果用户输入了错误的短选项,getopt 会打印一条消息:

  1. $ ./args -s -an file1 file2
  2. Suppress printing repeated blank lines
  3. ./args: invalid option -- 'a'
  4. What's that??
  5. Put line numbers next to all lines
  6. ------------------------------
  7. file1
  8. file2

下载速查表

getopt 还有更多的功能。例如,通过设计 -s string 或 -f file 这样的命令行语法规则,可以让短选项拥有自己的二级选项。你也可以告诉 getopt 在遇到无法识别的选项时不显示错误信息。使用 man 3 getopt 命令查看 getopt(3) 手册可以了解 getopt 的更多功能。

如果你需要 getopt() 和 getopt_long()的使用语法和结构上的提示,可以 下载我制作的速查表🔗 opensource.com。它提供了最小可行代码,并列出了你需要了解的一些全局变量的含义。速查表的一面是 getopt() 的用法,另一面是 getopt_long()的用法。


via: https://opensource.com/article/21/8/short-option-parsing-c

作者:Jim Hall 选题:lujun9972 译者:toknow-gh 校对:wxy

本文由 LCTT 原创编译,Linux中国 荣誉推出

LCTT 译者 :Xiangbin Ma
🌟🌟
翻译: 4.0 篇
|
贡献: 3165 天
2014-07-01
2023-03-01
https://linux.cn/lctt/toknow-gh
欢迎遵照 CC-BY-SA 协议规定转载,
如需转载,请在文章下留言 “转载:公众号名称”,
我们将为您添加白名单,授权“转载文章时可以修改”。


微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
如何在 Linux 中更新 Flatpak 软件包 | Linux 中国天赋“易昺(bǐng)”,创造历史!如何在 Linux 中降级 Flatpak 软件包 | Linux 中国5 个适合视力障碍者的 Linux 发行版 | Linux 中国Linux 实现简易的 Shell 命令行解释器通过 apt-get 降级一个软件包 | Linux 中国TUXEDO Stellaris 16(Gen5)是目前所能找到的终极 Linux 笔记本电脑 | Linux 中国如何在 Linux 系统中使用 Ventoy 创建多重引导的 U 盘 | Linux 中国使用 ChatGPT AI 从英文文本生成 Linux 命令 | Linux 中国Live Captions:Linux 上的开源视频字幕应用 | Linux 中国SpringBoot 应用的新命令行界面:Just希望的歌OBS Studio 29 发布,但对 Linux 用户来说变化不大 | Linux 中国纯命令行+美观UI,10款实用开源下载工具“狗得入内、中国人不得入内”应用层|关于编程语言中的那些「锁」事视频号——创作者的可选项还是必选项?回顾 2022 五个最佳 Linux 发行版 | Linux 中国在 Linux 中创建定时器 | Linux 中国如何在 Linux 中使用旧相机作为网络摄像头 | Linux 中国适用于 Linux 的五大流媒体直播应用 | Linux 中国何时起,“骰子”不读shǎi,改读tóu了?Why do some Chinese people live longer than many Americans?世界上只有两个 Linux 发行版:Arch Linux 与其它 | Linux 中国lnav: 用于 Linux 的高级日志文件浏览器 | Linux 中国Cinnamon 是一个被低估的神奇 Linux 桌面环境 | Linux 中国在 Linux 命令行上使用 dict 文字工具 | Linux 中国Garuda Linux “Raptor” 230305 版本发布 | Linux 中国新年愉快 轻奶酪蛋糕Linux查看进程占用网速和流量使用情况的三个命令:vnstat、iftop、nethogsRust 基础系列 #2: 在 Rust 程序中使用变量和常量 | Linux 中国如何在 Java 中使用方法 | Linux 中国美国档案---唐宝璜Agustín Hernández:中美洲建筑背景下的未来主义巨构Just:Spring Boot 应用的新命令行界面
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。