gets() 与 scanf() 函数相处呢有点小尴尬的,就是 gets() 在 scanf() 后边就爱捣乱。为什么呢,先了解它们两者之间的异同:
同: 都是可以接受连续的字符数据 并在字符结束后自动加上 ‘\0‘,标志结束接受
异:
scanf 不能接受空格、制表符Tab、回车等,遇空格时就结束接受
gets 则能够接受空格、制表符Tab和回车等,遇回车或EOF(end of file)时都会结束接受
当 gets() 在 scanf() 后,结束输入 scanf() 后回车时,gets()就把回车这个键给接收了。这关键就在于二者使用的结束标记不同。输入字符串时,scanf()遇到空格、回车、Tab结束,但在缓冲区中还留着这些结束符,此后如果使用gets()想去获取下一行字符串,它碰到的却是前面遗留下来的回车(或者回车之前还有空格等空白符),那么这次gets()就直接失效了。所以就出现了常碰到的第一个字符串变成空白字符串的现象。之前总是遇到,于是笔者本人就绕了个弯来解决这个问题。
本人所用的是 sscanf() 这个函数(是C语言中从一个字符串中读进与指定格式相符的数据的函数)来解决这个回车被吃掉的问题,关于 sscanf() 这个函数格式头文件 stdio.h 中这么描述:
_Check_return_ _CRT_INSECURE_DEPRECATE(sscanf_s)
_CRT_STDIO_INLINE int __CRTDECL sscanf(
_In_z_ char const* const _Buffer,
_In_z_ _Scanf_format_string_ char const* const _Format,
...)
sscanf(待读进字符串,指定与字符串相符的格式,变量数据列表)
举个粟子吧
#include<stdio.h>
int main(void)
{
int d;
char c, a[100] = "365hello", b[100];
sscanf(a, "%d%c%s", &d, &c, b);
printf("%d %c %s", d, c, b);
}
运行打印出结果 365 h ello ,这些都是符合指定的格式。也正是通过这样,我就同时用了两个 gets 第一个用来读取本就在 scanf()获取的字符信息,再通过 sscanf() 提取出可用的数据,这就有点麻烦了,不过对于特殊的输入还是有大用处的,因为 sscanf() 支持正则表达式。比如要求输入 m:4 而需要的数据是 4 ,那么就可以通过 sscanf() 正则表达式来将 : 忽略掉获取我们可用的数据。(关于强大的正则表达式可以参阅此文章 https://www.cnblogs.com/lanjianhappy/p/7171341.html)
后来,偶然的机会对尝试了将 scanf() 和 gets() 混用,不同的是用了 getchar() 在 scanf() 后,发现竟然可以,大概的原因是 getchar() 阻止了 gets() 吃掉 scanf() 的回车。(好像是废话。。。)再后者上网查阅了一下,原来还可以这样的,还有个方法就是在 scanf() 后加上个 scanf("\n") 算是再加上一个回车符来补充被 gets() 吃掉的回车吧。
像这样就对了,简单的可以用 scanf() 解决的就这样咯,同时也可适用 C++ 的 cin 不用像笔者那样来个 双gets() 来加上个 sscanf() 当然用用也无妨,特殊情况也是有的嘛。
#include<stdio.h>
int main(void)
{
int n;
char a[100];
scanf("%d", &n); //或这样 不用下一条的 scanf("\n"); 直接这样 scanf("%d\n", &n);
scanf("\n");
gets_s(a);
puts(a);
}