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

相关文章

SQLite3命令行工具最佳实践指南

《SQLite3命令行工具最佳实践指南》SQLite3是轻量级嵌入式数据库,无需服务器支持,具备ACID事务与跨平台特性,适用于小型项目和学习,sqlite3.exe作为命令行工具,支持SQL执行、数... 目录1. SQLite3简介和特点2. sqlite3.exe使用概述2.1 sqlite3.exe

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Go语言如何判断两张图片的相似度

《Go语言如何判断两张图片的相似度》这篇文章主要为大家详细介绍了Go语言如何中实现判断两张图片的相似度的两种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 在介绍技术细节前,我们先来看看图片对比在哪些场景下可以用得到:图片去重:自动删除重复图片,为存储空间"瘦身"。想象你是一个

Go语言中Recover机制的使用

《Go语言中Recover机制的使用》Go语言的recover机制通过defer函数捕获panic,实现异常恢复与程序稳定性,具有一定的参考价值,感兴趣的可以了解一下... 目录引言Recover 的基本概念基本代码示例简单的 Recover 示例嵌套函数中的 Recover项目场景中的应用Web 服务器中

Go语言中使用JWT进行身份验证的几种方式

《Go语言中使用JWT进行身份验证的几种方式》本文主要介绍了Go语言中使用JWT进行身份验证的几种方式,包括dgrijalva/jwt-go、golang-jwt/jwt、lestrrat-go/jw... 目录简介1. github.com/dgrijalva/jwt-go安装:使用示例:解释:2. gi

Go 语言中的 Struct Tag 的用法详解

《Go语言中的StructTag的用法详解》在Go语言中,结构体字段标签(StructTag)是一种用于给字段添加元信息(metadata)的机制,常用于序列化(如JSON、XML)、ORM映... 目录一、结构体标签的基本语法二、json:"token"的具体含义三、常见的标签格式变体四、使用示例五、使用

Go语言使用slices包轻松实现排序功能

《Go语言使用slices包轻松实现排序功能》在Go语言开发中,对数据进行排序是常见的需求,Go1.18版本引入的slices包提供了简洁高效的排序解决方案,支持内置类型和用户自定义类型的排序操作,本... 目录一、内置类型排序:字符串与整数的应用1. 字符串切片排序2. 整数切片排序二、检查切片排序状态:

一文详解PostgreSQL复制参数

《一文详解PostgreSQL复制参数》PostgreSQL作为一款功能强大的开源关系型数据库,其复制功能对于构建高可用性系统至关重要,本文给大家详细介绍了PostgreSQL的复制参数,需要的朋友可... 目录一、复制参数基础概念二、核心复制参数深度解析1. max_wal_seChina编程nders:WAL

基于Go语言实现Base62编码的三种方式以及对比分析

《基于Go语言实现Base62编码的三种方式以及对比分析》Base62编码是一种在字符编码中使用62个字符的编码方式,在计算机科学中,,Go语言是一种静态类型、编译型语言,它由Google开发并开源,... 目录一、标准库现状与解决方案1. 标准库对比表2. 解决方案完整实现代码(含边界处理)二、关键实现细

如何合理管控Java语言的异常

《如何合理管控Java语言的异常》:本文主要介绍如何合理管控Java语言的异常问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍2、Thorwable类3、Error4、Exception类4.1、检查异常4.2、运行时异常5、处理方式5.1. 捕获异常