问答题 函数ReadDat()的功能是从文件ENG.IN中读取一篇英文文章,存入字符串数组xx中。请编写函数encryptChar(),按给定的替代关系对数组xx中的所有字符进行替换,替换后的字符仍存入数组xx的对应的位置上,最后调用函数WriteDat(),把结果xx输出到文件PS.DAT中。
替代关系:f(p)=p*11 mod 256(p是数组xx中某一个字符的ASCII码值,f(p)是计算后新字符的ASCII码值),如果原字符的ASCII值是偶数或计算后f(p)的值小于等于32,则该字符不变,否则,将f(p)所对应的字符进行替换。
注意:原始数据文件存放的格式是:每行的宽度均小于80个字符。请勿改动主函数main()、读函数ReadDat()和写函数WriteDat()的内容。
【试题程序】
#include < stdio.h >
#include < string.h >
#include < stdlib.h >
#include < ctype.h >
unsigned char xx[50][80];
int maxline =0; //文章的总行数
int ReadDat(void);
void WriteDat(void);
void encryptChar()
{
}
void main()
{
system("CLS");
if (Readmat ())
{
printf ("数据文件 ENG. IN 无法打开!
\n\007");
return;
}
encryptChar();
WriteDat();
}
int ReadDat (void)
{
FILE * fp;
int i =0;
unsigned char * p;
if ((fp = fopen ("ENG. IN","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;
}
void WriteDat(void)
{
FILE * fp;
int i;
fp = fopen ("PS.DAT", "w") ;
for(i=0; i<maxline; i++)
{
printf("%s\n", xx[i]);
fprintf(fp, "%s\n", xx[i]);
}
fclose(fp);
}
【正确答案】
【答案解析】void encryptChar()
{
int i,j; /*定义循环控制变量*/
int str; /*存储字符串的长度*/
char ch; /*存储当前取得的字符*/
for(i=0;i<maxline;i++) /*以行为单位获取字符*/
{
str=strlen(xx[i]); /*求得当前行的字符串长度*/
for(j=0;j<str;j++) /*依次取每行的所有字符*/
{
ch=xx[i][j]*11%256;
if(xx[i][j]%2==0 || ch<=32)
continue; /*如果原字符的ASCII值是偶数或计算后的值小于等于32,则该字符不变*/
else
xx[i][j]=ch; /*否则将所对应的字符进行替代*/
}
}
} [考点] 本题考查对字符数组中字符计算和替换。考查的知识点主要包括:字符串数组的访问,字符ASCII码的算术运算,if判断结构以及逻辑表达式。
[解析] 此题属于字符替代问题;分析题干要求,可以归纳出3个关键点:关键点1如何对字符数组的元素逐个访问;关键点2如何根据给出的函数替代关系"f(p)=p*11 mod 256"对字符进行计算;关键点3根据条件(本题为"原字符的ASCII值是偶数或计算后f(p)的值小于等于32")对计算结果进行判断,并分别对满足与不满足条件的情况进行处理。
通过问题分析,得出解此题的思路为:首先通过字符串处理函数strlen获取字符串的长度,根据获得的长度使用下标法逐一对字符数组的元素进行访问;然后按照题目给出的函数关系式直接对字符进行算术运算;最后通过if判断结构和逻辑表达式判断计算结果是否满足条件,分别对两种情况进行处理。
根据函数替代关系对字符进行运算,if判断结构中逻辑表达式。