C语言KR圣经笔记 5.10命令行参数

2024-01-19 13:44

本文主要是介绍C语言KR圣经笔记 5.10命令行参数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 5.10 命令行参数

在支持 C 语言的环境中,有一种方法可以在程序开始执行时将命令行参数传递给程序。当 main 被调用时,会带着两个参数。第一个是程序被调用时带的命令行参数个数(按惯例称为 argc,即参数个数 argument count 的缩写),第二个是指向包含所有参数的字符串数组的指针(argv,参数向量 argument vector 的缩写),数组里每个字符串对应一个参数。我们习惯使用多级指针来操作这些字符串。

最简单的一个例子是程序 echo,echo 会将它的命令行参数回显到一行内,参数之间用空格分隔。也就是说,如下命令

echo hello, world

会打印

hello, world

根据惯例,argv[0] 是被调用程序的名字,因此 argc 至少为 1。若 argc 为1,说明在程序名之后没有命令行参数。在上面的例子中 argc 为 3,而argv[0],argv[1],argv[2] 分别是 "echo", "hello," 和 "world"。第一个可能存在的参数是 argv[1],最后一个是 argv[argc -1],另外,C 标准要求 argv[argc] 必须是空指针。

echo 的第一个版本将 argv 作为字符指针的数组来对待

#include <stdio.h>/* 回显命令行参数;第一版 */
main(int argc, char *argv[])
{int i;for (i = 1; i < argc; i++)printf("%s%s", argv[i], (i < argc-1) ? " " : "");printf("\n");return 0;
}

既然 argv 是指向指针数组的指针,我们可以操作指针而不是数组索引。下面这个版本在 argc 递减时,对指向 char 指针的指针 argv 进行递增。

#include <stdio.h>/* 回显命令行参数;第二版 */
main(int argc, char *argv[])
{while (--argc > 0)printf("%s%s", *++argv, (argc > 1) ? " " : "")printf("\n");return 0;
}

由于 argv 是指向参数字符串数组开头的指针,将其递增 1 (++argv)使它一开始就指向 argv[1] 而不是 argv[0]。后续每次递增将其移到下一个参数;然后 *argv 就是该参数的指针。同时,argc递减,当它变为零时,就没有待打印的参数了。

另外,我们也可以把 printf 写成

    printf((argc > 1) ? "%s " : "%s", *++argv);

这表示 printf 的格式化参数也可以是一个表达式。

第二个例子,是对 4.1 节的样式搜索程序做一些增强。如果你还能记得的话,我们把要搜索的样式深埋在了程序里面,这种做法明显没法让人满意。我们参考 UNIX 程序 grep 来修改这个程序,使要匹配的样式通过命令行的第一个参数来指定。

#include <stdio.h>
#include <string.h>
#define MAXLINE 1000int getline(char *line, int max);/* find:打印匹配第一个参数的行 */
main(int argc, char *argv[])
{char line[MAXLINE];int found = 0;if (argc != 2)printf("Usage: find pattern\n");elsewhile (getline(line, MAXLINE) > 0)if (strstr(line, argv[1]) != NULL) {printf("%s", line);found++;}return found;
}

标准库函数 strstr(s, t) 返回字符串 t 在字符串 s 中首次出现的位置,若 s 不包含 t 则返回 NULL。该函数在 <string.h> 中声明。

现在可以对上面的原型程序进行完善,以此来说明更多的指针结构。假定我们想增加两个可选参数。一个是“打印除了匹配样式之外的所有行”;第二个是“在每个输出的行前面加上行号”。

UNIX 系统里 C 程序的通用惯例,是用负号开头的参数表示一个可选的标志符或参数。如果我们选用 -x (代表“除了” except)来表示取反,用 -n (行数 “number”)来要求输出行编号,则如下命令

find -x -n pattern

将打印每个不匹配样式的行,每行前面有行号。

应当允许可选的参数以任意的顺序出现,而程序的其他部分应当独立于给出的参数个数。更进一步,如果选项参数可以结合起来,对用户会更方便,如

find -nx pattern

这里是完善后的程序:

#include <stdio.h>
#include <string.h>
#define MAXLINE 1000int getline(char *line, int max);/* find:打印与第一个参数匹配的行 */
main(int argc, char *argv[])
{char line[MAXLINE];long lineno = 0;int c, except = 0, number = 0, found = 0;while (--argc > 0 && (*++argv)[0] == '-')while (c = *++argv[0])switch (c) {case 'x':except = 1;break;case 'n':number = 1;break;default:printf("find: illegal option %c\n", c);argc = 0;found = -1;break;}if (argc != 1)printf("Usage: find -x -n pattern\n");elsewhile (getline(line, MAXLINE) > 0) {lineno++;if ((strstr(line, *argv) != NULL)) != except) {if (number)printf("%ld:", lineno);printf("%s", line);found++;}}return found;
}

在每个可选参数之前,argc 递减而 argv 递增。在循环结束时,如果没有错误,则 argc 告诉我们还剩多少参数未处理,而 argv 指向这些剩余参数中的第一个。因此 argc 应当为 1 而 *argv 应指向要匹配的样式。注意 *++argv 是指向参数字符串的指针,因此 (*++argv)[0] 是其首个字符。(另一种合法的写法是 **++argv)。由于 [ ] 比 * 和 ++ 绑定得更紧,因此括号是必须的;若没有括号,表达式会变为 *++(argv[0])。实际上,这正是我们在内层循环中所使用的,这里的任务是在特定的参数字符串中遍历。在内层循环中,表达式 *++argv[0] 对指针 argv[0] 进行递增!

比上面这些指针表达式还复杂的使用场景是很稀少的;在那些情况下,将其拆为两个或三个步骤会更直观。

练习5-10、写一个程序 expr,计算从命令行输入的逆波兰表达式,其中每个操作符或操作数是一个单独的参数。例如  expr 2 3 4 + * 会对 2 * (3+4) 求值。

练习5-11、扩展(练习1-21中的) entab 和 detab 以支持简写, entab -m +n, 表示制表符从第 m 列开始,每 n 列停止。选择(对用户来说)方便的默认行为。

练习5-13、写程序 tail,输出其输入的末尾 n 行。我们定义默认 n 为 10,但可以通过可选参数进行修改,例如 tail -n 打印最后 n 行。不管输入或者 n 的值多么不合理,程序都应该有合理的行为。程序应当最大化地利用存储空间;文本行应当像 5.6 节中的排序程序一样存储,而不能存为固定长度的二维数组。

这篇关于C语言KR圣经笔记 5.10命令行参数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Andrej Karpathy最新采访:认知核心模型10亿参数就够了,AI会打破教育不公的僵局

夕小瑶科技说 原创  作者 | 海野 AI圈子的红人,AI大神Andrej Karpathy,曾是OpenAI联合创始人之一,特斯拉AI总监。上一次的动态是官宣创办一家名为 Eureka Labs 的人工智能+教育公司 ,宣布将长期致力于AI原生教育。 近日,Andrej Karpathy接受了No Priors(投资博客)的采访,与硅谷知名投资人 Sara Guo 和 Elad G

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

如何在页面调用utility bar并传递参数至lwc组件

1.在app的utility item中添加lwc组件: 2.调用utility bar api的方式有两种: 方法一,通过lwc调用: import {LightningElement,api ,wire } from 'lwc';import { publish, MessageContext } from 'lightning/messageService';import Ca

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

4B参数秒杀GPT-3.5:MiniCPM 3.0惊艳登场!

​ 面壁智能 在 AI 的世界里,总有那么几个时刻让人惊叹不已。面壁智能推出的 MiniCPM 3.0,这个仅有4B参数的"小钢炮",正在以惊人的实力挑战着 GPT-3.5 这个曾经的AI巨人。 MiniCPM 3.0 MiniCPM 3.0 MiniCPM 3.0 目前的主要功能有: 长上下文功能:原生支持 32k 上下文长度,性能完美。我们引入了

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

用命令行的方式启动.netcore webapi

用命令行的方式启动.netcore web项目 进入指定的项目文件夹,比如我发布后的代码放在下面文件夹中 在此地址栏中输入“cmd”,打开命令提示符,进入到发布代码目录 命令行启动.netcore项目的命令为:  dotnet 项目启动文件.dll --urls="http://*:对外端口" --ip="本机ip" --port=项目内部端口 例: dotnet Imagine.M

AI(文生语音)-TTS 技术线路探索学习:从拼接式参数化方法到Tacotron端到端输出

AI(文生语音)-TTS 技术线路探索学习:从拼接式参数化方法到Tacotron端到端输出 在数字化时代,文本到语音(Text-to-Speech, TTS)技术已成为人机交互的关键桥梁,无论是为视障人士提供辅助阅读,还是为智能助手注入声音的灵魂,TTS 技术都扮演着至关重要的角色。从最初的拼接式方法到参数化技术,再到现今的深度学习解决方案,TTS 技术经历了一段长足的进步。这篇文章将带您穿越时