1.1.4 read_column_numbers函数
/* |
这几行构成了read_column_numbers函数的起始部分。注意,这个声明和早先出现在程序中的该函数原型的参数个数和类型以及函数的返回值完全匹配。如果出现不匹配的情况,编译器就会报错。
在函数声明的数组参数中,并未指定数组的长度。这种格式是正确的,因为不论调用函数的程序传递给它的数组参数的长度是多少,这个函数都将照收不误。这是一个伟大的特性,它允许单个函数操纵任意长度的一维数组。这个特性不利的一面是函数没法知道该数组的长度。如果确实需要数组的长度,它的值必须作为一个单独的参数传递给函数。
当本例的read_column_numbers函数被调用时,传递给函数的其中一个参数的名字碰巧与上面给出的形参名字相同。但是,其余几个参数的名字与对应的形参名字并不相同。和绝大多数语言一样,C语言中形式参数的名字和实际参数的名字并没有什么关系。你可以让两者相同,但这并非必须。
int num = 0; |
/* |
这又是一个循环,用于读取列标号。scanf函数从标准输入读取字符并根据格式字符串对它们进行转换——类似于printf函数的逆操作。scanf函数接受几个参数,其中第1个参数是一个格式字符串,用于描述期望的输入类型。剩余几个参数都是变量,用于存储函数所读取的输入数据。scanf函数的返回值是函数成功转换并存储于参数中的值的个数。
警告:
对于这个函数,你必须小心在意,理由有二。首先,由于scanf函数的实现原理,所有标量参数的前面必须加上一个“&”符号。关于这点,第8章我会解释清楚。数组参数前面不需要加上“&”符号 。但是,数组参数中如果出现了下标引用,也就是说实际参数是数组的某个特定元素,那么它的前面也必须加上“&”符号。在第15章,我会解释在标量参数前面加上“&”符号的必要性。现在,你只要知道必须加上这个符号就行了,因为如果没有它们的话,程序就无法正确运行。
警告:
第二个需要注意的地方是格式代码,它与printf函数的格式代码颇为相似却又并不完全相同,所以很容易引起混淆。表1.2粗略列出了一些你可能会在scanf函数中用到的格式代码。注意,前5个格式代码用于读取标量值,所以变量参数的前面必须加上“&”符号。使用所有格式码(除了%c之外)时,输入值之前的空白(空格、制表符、换行符等)会被跳过,值后面的空白表示该值的结束。因此,用%s格式码输入字符串时,中间不能包含空白。除了表中所列之外,还存在许多格式代码,但这张表里面的这几个格式代码对于应付我们现在的需求已经足够了。
我们现在可以解释表达式:
scanf("%d", &columns[num] ) |
格式码%d表示需要读取一个整型值。字符是从标准输入读取,前导空白将被跳过。然后这些数字被转换为一个整数,结果存储于指定的数组元素中。我们需要在参数前加上一个“&”符号,因为数组下标选择的是一个单一的数组元素,它是一个标量。
while循环的测试条件由3个部分组成:
num < max |
columns[num] >= 0 |
这个表达式确保函数所读取的值是正数。如果两个测试条件之一的值为假,循环就会终止。
表1.2 常用scanf格式码
|
格 式 |
含 义 |
变 量 类 型 |
|
%d |
读取一个整型值 |
int |
|
%ld |
读取一个长整型值 |
long |
|
%f |
读取一个实型值(浮点数) |
float |
|
%lf |
读取一个双精度实型值 |
double |
|
%c |
读取一个字符 |
char |
|
%s |
从输入中读取一个字符串 |
char型数组 |
1.但是,即使你在它前面加上一个“&”也没有什么不对,所以如果你喜欢,也可以加上它。
提示:
标准并未硬性规定C编译器对数组下标的有效性进行检查,而且绝大多数C编译器确实也不进行检查。因此,如果你需要进行数组下标的有效性检查,你必须自行编写代码。如果此处不进行num < max这个测试,而且程序所读取的文件包含超过20个列标号,那么多出来的值就会存储在紧随数组之后的内存位置,这样就会破坏原先存储在这个位置的数据,可能是其他变量,也可以是函数的返回地址。这可能会导致多种结果,程序很可能不会按照你预想的那样运行。
&&是“逻辑与”操作符。要使整个表达式为真,&&操作符两边的表达式都必须为真。然而,如果左边的表达式为假,右边的表达式便不再进行求值,因为不管它是真是假,整个表达式总是假的。在这个例子中,如果num到达了它的最大值,循环就会终止 ,而表达式
columns[num] |
警告:
此处需要小心。当你实际上想使用&&操作符时,千万不要误用了&操作符。&操作符执行“按位与”的操作,虽然有些时候它的操作结果和&&操作符相同,但很多情况下都不一样。我将在第5章讨论这些操作符。
scanf函数每次调用时都从标准输入读取一个十进制整数。如果转换失败,不管是因为文件已经读完还是因为下一次输入的字符无法转换为整数,函数都会返回0,这样就会使整个循环终止。如果输入的字符可以合法地转换为整数,那么这个值就会转换为二进制数存储于数组元素columns[num]中。然后,scanf函数返回1。
警告:
注意:用于测试两个表达式是否相等的操作符是==。如果误用了=操作符,虽然它也是合法的表达式,但其结果几乎肯定和你的本意不一样:它将执行赋值操作而不是比较操作!但由于它也是一个合法的表达式,所以编译器无法为你找出这个错误 。在进行比较操作时,千万要注意你所使用的是两个等号的比较操作符。如果你的程序无法运行,请检查一下所有的比较操作符,看看是不是这个地方出了问题。相信我,你肯定会犯这个错误,而且可能不止一次,我自己就曾经犯过这个错误。
接下来的一个&&操作符确保在scanf函数成功读取了一个数之后才对这个数进行是否赋值的测试。语句
num += 1; |
num = num + 1; |
/* |
这个测试检查程序所读取的整数是否为偶数个,这是程序规定的,因为这些数字要求成对出现。%操作符执行整数的除法,但它给出的结果是除法的余数而不是商。如果num不是一个偶数,它除以2之后的余数将不是0。
puts函数是gets函数的输出版本,它把指定的字符串写到标准输出并在末尾添上一个换行符。程序接着调用exit函数,终止程序的运行,EXIT_FAILURE这个值被返回给操作系统,提示出现了错误。
/* |
当scanf函数对输入值进行转换时,它只读取需要读取的字符。这样,该输入行包含了最后一个值的剩余部分仍会留在那里,等待被读取。它可能只包含作为终止符的换行符,也可能包含其他字符。不论如何,while循环将读取并丢弃这些剩余的字符,防止它们被解释为第1行数据。
下面这个表达式
(ch = getchar() ) != EOF && ch != '\n' |
值得花点时间讨论。首先,getchar函数从标准输入读取一个字符并返回它的值。如果输入中不再存在任何字符,函数就会返回常量EOF(在stdio.h中定义),用于提示文件的结尾。
从getchar函数返回的值被赋给变量ch,然后把它与EOF进行比较。在赋值表达式两端加上括号用于确保赋值操作先于比较操作进行。如果ch等于EOF,整个表达式的值就为假,循环将终止。若非如此,再把ch与换行符进行比较,如果两者相等,循环也将终止。因此,只有当输入尚未到达文件尾并且输入的字符并非换行符时,表达式的值才是真的(循环将继续执行)。这样,这个循环就能剔除当前输入行最后的剩余字符。
现在让我们进入有趣的部分。在大多数其他语言中,我们将像下面这个样子编写循环:
ch = getchar(); |
ch = getchar(); |
C可以把赋值操作蕴含于while语句内部,这样就允许程序员消除冗余语句。
提示:
例子程序中的那个循环的功能和上面这个循环相同,但它包含的语句要少一些。无可争议,这种形式可读性差一点。仅仅根据这个理由,你就可以理直气壮地声称这种编码技巧应该避免使用。但是,你之所以会觉得这种形式的代码可读性较差,只是因为你对C语言及其编程的习惯用法不熟悉之故。经验丰富的C程序员在阅读(和编写)这类语句时根本不会出现困难。在没有明显的好处时,你应该避免使用影响代码可读性的方法。但在这种编程习惯用法中,同样的语句少写一次带来的维护方面的好处要更大一些。
一个经常问到的问题是:为什么ch被声明为整型,而我们事实上需要它来读取字符?答案是EOF是一个整型值,它的位数比字符类型要多,把ch声明为整型可以防止从输入读取的字符意外地被解释为EOF。但同时,这也意味着接收字符的ch必须足够大,足以容纳EOF,这就是ch使用整型值的原因。正如第3章所讨论的那样,字符只是小整型数而已,所以用一个整型变量容纳字符值并不会引起任何问题。
提示:
对这段程序最后还有一点说明:这个while循环的循环体没有任何语句。仅仅完成while表达式的测试部分就足以达到我们的目的,所以循环体就无事可干。你偶尔也会遇到这类循环,处理它们应该没问题。while语句之后的单独一个分号称为空语句(empty statement),它就是应用于目前这个场合,也就是语法要求这个地方出现一条语句但又无需执行任何任务的时候。这个分号独占一行,这是为了防止读者错误地以为接下来的语句也是循环体的一部分。
return num; |
| 回书目 上一节 下一节 |