程序示例
/*
** 这个程序从标准输入中读取输入行并在标准输出中打印这些输入行
** 每个输入行的后面一行是该行内容的一部分
**
** 输入的第一行是一串列表号,串的是最后以一个负数结尾
** 这些列标号成对出现,说明需要打印的输入行的列的范围
** 例如,0 3 10 12 -1 表示第0列到第3列,第10列到第12列的内容将被打印
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX_COLS 20 /* 所能处理的最大列号 */
#define MAX_INPUT 1000 /* 每个输入行的最大长度 */
int read_column_numbers(int columns[], int max);
void rearrange(char *output, char const *input,
int n_columns, int const columns[]);
int main(void){
int n_columns; /* 进行处理的列标号 */
int columns[MAX_COLS]; /* 需要处理的列数 */
char input[MAX_INPUT]; /* 容纳输入行的数组 */
char output[MAX_INPUT]; /* 容纳输出行的数组 */
/*
** 读取该串列标号
*/
n_columns = read_column_numbers(columns, MAX_COLS);
/*
** 读取 处理 和 打印剩余的输入行
*/
while(gets(input) != NULL){
printf("Original input: %s\n",input);
rearrange(output, input, n_columns, columns);
printf("Rearranged line: %s\n", output);
}
}
//读取序列号,如果超出范围则不给予理会
int read_column_numbers(int columns[],int max){
int num = 0;
int ch;
//取得序列号,如果读取的序列号小于0则停止读取
while (num < max && scanf("%d",&columns[num]) == 1 && columns[num] >= 0)
num ++;
//确认已经读取的序列号为偶数个,因为它们是成对出现的
if (num%2 != 0){
puts("Last column number is not paired.");
exit(EXIT_FAILURE);
}
//丢弃该行中包含最后一个数字的那部分内容
//比如当出现-1后,如果后面还有内容的话,后面的内容应该处理,否则会变为输入字符串内容
while ((ch = getchar())!= EOF && ch != '\n' );
return num;
}
//处理输入行,将指定列的字符连在一起,输出行以NUL结尾
void rearrange(char *output, char const *input, int n_columns, int const columns[]){
int col; /* columns数组的下标 */
int output_col; /* 输出列计数器 */
int len; /* 输入行的长度 */
len = strlen(input);
output_col = 0;
//处理每对列标号
for (col = 0; col < n_columns; col += 2){
int nchars = columns[col+1] - columns[col] + 1;
//如果输入行结束或输出行数组已满,就结束任务
if (columns[col] >= len || output_col == MAX_INPUT-1)
break;
//如果输出行数据空间不够,只复制可以容纳的数据
if (output_col + nchars > MAX_INPUT-1)
nchars = MAX_INPUT - output_col -1;
//复制相关数据
strncpy(output+output_col,input+columns[col],nchars);
output_col += nchars;
}
output[output_col] = '\0';
}
预处理指令
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX_COLS 20 /* 所能处理的最大列号 */
#define MAX_INPUT 1000 /* 每个输入行的最大长度 */
以上5行都为预处理指令。因为它们是由预处理器解释的。预处理器读入源代码,根据预处理指令对其进行修改,然后将把修改后的源代码递交给编译器。
函数原型
int read_column_numbers(int columns[], int max);
void rearrange(char *output, char const *input,
int n_columns, int const columns[]);
这些声明被称为函数原型。它们告诉编译器这些以后将在源文件中定义的函数的特征。这样,当这些函数被调用时,编译器就能对它们进行准确性检查。
参数传递
在C语言中,数组参数是以引用形式进行传递的,也就是传递调用,而标量和常量则是按值传递的。
- 注意
在函数中对标量参数的任何修改都会在函数返回时丢失,因此,被调用函数无法修改调用函数以传值形式传递给它的参数。然而,当被调用函数修改数组参数的其中一个参数时,调用函数所传递的数组会被实际地修改。
get函数
while(gets(input) != NULL){
printf("Original input: %s\n",input);
rearrange(output, input, n_columns, columns);
printf("Rearranged line: %s\n", output);
}
get函数从标准输入读取一行文本并把它存储于作为参数传递给它的数组中。一行输入由一串字符组成,以一个换行符结尾。gets函数丢弃换行符,并在该行的末尾存储一个NUL字节。
字符串常量
字符串常量就是源程序中被双引号括起来的一串字符。例如,“Hello”,在内存中占据6个字节的空间,按顺序分别是H、e、l、l、o和NUL。
printf函数
printf("Original input: %s\n",input);
scanf函数
while (num < max && scanf("%d",&columns[num]) == 1 && columns[num] >= 0)
scanf函数从标准输入读取字符并根据格式字符串对它们进行转换。scanf函数的返回值是函数转换成功并存储于参数中的值的个数。
- 注意
所有的标量参数的前面必须加上一个“&”符号,而对于数组参数,可加可不加。但如果是数组中的一个参数,则必须加。
puts("Last column number is not paired.");
put函数是gets函数的输出版本,它把指定的字符串写到标准输出并在末尾添上一个换行符。
int ch;
while ((ch = getchar())!= EOF && ch != '\n' );
getchar函数从标准输入流读取一个字符并返回它的值。如果输入中不再存在任何字符,函数就会返回常量EOF(在stdio.h中定义),用于提示文件的结尾。
- 为什么这里ch被声明为整型?
EOF是一个整型值,它的位数比字符类型要多,把ch声明为整型可以防止从输入读取的字符意外被解释为EOF。但同时,这也意味着接收字符的ch必须足够大,足以容纳EOF,这就是ch使用整型值的原因。实际上,字符只是小整型数,所以用一个整型变量容纳字符值不会引起任何变化。
const类型
void rearrange(char *output, char const *input, int n_columns, int const columns[]){
int col; /* columns数组的下标 */
int output_col; /* 输出列计数器 */
int len; /* 输入行的长度 */
当数组名作为实参时,传给函数的实际上是一个指向数组起始位置的指针,也就是数组在内存中的地址。将columns声明为const有两方面的作用:首先,它声明该函数的作者的意图是这个参数不能被修改。其次,它导致编译器去验证是否违背该意图。
strncpy函数
strncpy(output+output_col,input+columns[col],nchars);
strncpy函数的前两个参数分别是目标字符串和源字符串的地址,最后一个参数指定需要复制的字符数。