题目如下:
本来遇到一些问题,还好都解决了。这里记录一下:
0x01 回顾gets、fgets、scanf函数的使用和区别
1 #include2 #include <string.h> 3 4 int main(void) 5 { 6 char buffer[201]; 7 char *ptr; 8 ptr = gets(buffer); 9 printf("buffer length = %d, or %d, buffer = '%s',ptr = '%s'.\n",strlen(buffer),strlen(ptr),buffer,ptr);
10 return 0; 11 }
我们输入的字符串为'hello world!',共12个字符,完整的显示出来;值得注意的是gets函数返回值是一个字符串指针,指向输入的字符串,也就是buffer数组的首地址。
然而可能会出现这样一种问题,gets(buffer);的buffer数组大小有限,但是从控制台输入的信息可能会超过这个数组大小,这个函数就会轻易的导致缓冲区溢出问题且难以避免。因此在系统安全性更高的Linux gcc环境下编译,带有gets函数的程序被编译时会出现一些警告,甚至出错;它会提示你使用更安全的fgets函数来替代gets函数减少意料之外的错误。
1 int main(void) 2 { 3 char buffer[65]; 4 char *ptr; 5 ptr = fgets(buffer,sizeof(buffer),stdin); 6 printf("buffer length = %d, or %d, buffer = '%s',ptr = '%s'.\n",strlen(buffer),strlen(ptr),buffer,ptr); 7 return 0; 8 }
我们输入的字符串为'hello world!',明明只有12个字符,却显示字符串长度竟然为13,打印字符串竟然意外换行(单引号另起一行可以看出),可见fgets函数和gets函数区别在于fgets可以读取最后的换行符为结尾使得buffer = “hello world!\n\0”,长度包括\n所占的一个字符。
其次,值得注意的是fgets函数控制输入字符的最大数目(fgets接受的第二个参数是输入字符的最大数目),这和gets函数相比具有较高的安全性,避免了缓冲区溢出的可能性:因为我把参数2设为sizeof(buffer)就完全不用担心输入长度超过buffer大小带来的溢出风险。而且,fgets函数第三个参数指定了从哪里读入一行字符(到换行符\n结束读),可以从打开的文件指针fp读出一行,也可以使用stdin从控制端输入。
1 int main() 2 { 3 char buffer[65]; 4 scanf("%s",buffer); 5 printf("buffer length = %d ,buffer = '%s'.\n",strlen(buffer),buffer); 6 return 0; 7 }
能看出来,scanf函数读取字符串时候遇到空格或者换行符就停止读取(字符串末尾不包含换行符或最后的空格),依次不适合用来读取带空格的长字符串。但是当已知输入字符串每行格式为固定形式的时候,可以定义好scanf格式化参数来分别接收分隔开的参数。例如学籍信息输入学生信息记录为:学生姓名 学号 性别 专业 班级 ,拿上个例子继续优化scanf如下:
1 int main() 2 { 3 char buffer[65],buffer2[65]; 4 scanf("%s %s",buffer,buffer2); 5 printf("buffer length = %d ,buffer2 length = %d ,buffer = '%s',buffer2 = '%s'.\n",strlen(buffer),strlen(buffer2),buffer,buffer2); 6 return 0; 7 }
0x02 题目的解
先定义两个函数,一个函数根据输入的字符,对照键盘,得到打印字符需要的次数。另一个函数接受字符串参数,每个字符都调用第一个函数得到数字,值加和并返回直到末尾字符 '#' 。
1 int char_conv(char c) 2 { 3 if(c == 'a'||c == 'd'||c == 'g'||c == 'j'|| c == 'm'||c == 'p'||c == 't'||c == 'w'||c == ' ') 4 return 1; 5 else if(c == 'b'||c == 'e'||c == 'h'||c == 'k'||c == 'n'||c == 'q'||c == 'u'||c == 'x') 6 return 2; 7 else if(c == 'c'||c == 'f'||c == 'i'||c == 'l'||c == 'o'||c == 'r'||c == 'v'||c == 'y') 8 return 3; 9 else if(c == 's'||c == 'z') 10 return 4; 11 else 12 return 0; 13 } 14 15 int string_conv(const char *src , int length) 16 { 17 char temp='\0'; 18 int index = 0,mark=0; 19 while(index < length) 20 { 21 temp = src[index]; 22 if(temp == '#') 23 break; 24 mark += char_conv(temp); 25 index++; 26 } 27 return mark; 28 }
其次,确定主函数结构,使用gets函数接受字符串,并保存指针地址为ptr,调用已定义的函数计算出每个字符串的键盘敲击次数存入数组array;最后遍历array并打印:
1 int main() 2 { 3 int N=0,i=0; 4 char buffer[201],*ptr; 5 int array[10]; //假设最多输出10个字符串 6 7 8 scanf("%d",&N); 9 getchar(); //这行出乎人意料的重要,你可以注释掉这一行看看执行效果…… 10 for(i=0;i) 11 { 12 ptr = gets(buffer); 13 array[i] = string_conv(ptr,strlen(buffer)); 14 memset(buffer,0,sizeof(buffer)); 15 } 16 i = 0; 17 while(i<N) 18 { 19 printf("%d\n",array[i]); 20 i++; 21 } 22 return 0; 23 }
运行结果: