问答题 函数ReadDat()实现从文件in.dat中读取一篇英文文章,存入到无符号字符串数组xx中;请编制函数eneryptChar(),按给定的替代关系对数组xx中的所有字符进行替代,其替代值仍存入数组xx所对应的位置上,最后调用函数WriteDac()把结果xx输出到文件out.dat中。
替代关系:f(p)=p*11 mod 256, mod为取余运算,p是数组xx中某一个字符的ASCII值,f(p)是计算后的无符号整型值(注意:中间变量定义成整型变量),如果计算后f(p)值小于等于32或大于130,则该字符不变,否则用f(p)替代所对应的字符。
请勿改动主函数main()、读数据函数ReadDat()和输出数据函数WfiteDat()的内容。
#include <stdio.h>
#include <string.h>
#include <ctype.h>
unsigned char xx[50][80];
int maxline=0; /*文章的总行数*/
int ReadDat (void);
void WriteDat (void);
void encryptChar()
{
}
void main()
{
if(ReadDat())
{
printf("数据文件in.dat不能打开!\n\007");
return;
}
encryptChar();
WriteDat();
}
/*从文件in.dat中读取一篇英文文章, 存入到无符号字符串数组xx中*/
int ReadDat (void)
{
FILE *fp;
int i=0;
unsigned char *p; //定义无符号字符型指针变量p
if((fp=fopen("in.dat", "r"))==NULL)
return 1;
while (fgets(xx[i], 80, fp)!=NULL)
{
p=strchr (xx[i], "\n");
if(p) *p=0;
i++;
}
maxline=i;
fclose(fp);
return 0;
}
/*把结果xx输出到文件out.dat中*/
void WriteDat (void)
{
FILE *fp;
int i;
fp=fopen("out.dat", "w");
for(i=0; i<maxline; i++)
{
printf("%s\n", xx[i]);
fprintf(fp, "%s\n", xx[i]);
}
fclose (fp);
}
【正确答案】
【答案解析】int i, j, val;
/*对字符串数组xx中的每一个字符,用替代公式计算其对应的ASCII值,如果ASCII值小于等于32或大于130,则该字符不变,否则用新的ASCII值对应的字符替代所原有的字符*/
for(i=0; i<maxline; i++)
for(j=0; j<(int) strlen(xx[i]); j++)
{
val=(xx[i][j]*11)%256; //xx[i][j]对应的ASCII值乘以11然后再对256求余, 得到其对应的新的ASCII值
if(!(val<=32||val>130)) //如果新的对应的ASCII值大于32且小于等于130时
xx[i][j]=val; //用新的ASCII值对应的字符替代原来的字符
} [解析] 字符替代f(p)=p*11 mod256,小于等于32或大于130不变,否则替换。
①首先定义两个循环整型变量i,j和存放ASCII码值的中间变量val。
②通过两个for循环完成对字符串的替换。第一个for循环的循环变量i的初值为0,当i值等于文章中最大行数值maxline时停止循环,每循环一次,i的值加1;第二层for循环中,循环变量j的初值为0,表示从第i行的第1个字符开始计算,当j值等于该行中字符串长度的数值时停止循环,每循环一次,j的值加1。
③循环体中,根据题意重新计算每个字符xx[i][j]的ASCII码值,当新取得的ASCII码值大于32且小于等于130时,则用新的ASCII码值所对应的字符替代原来的字符,新ASCII码值不在此范围内的字符则保持原字符不变。