欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

gets() 与 scanf() 的小尴尬

程序员文章站 2022-05-18 08:50:49
gets() 与 scanf() 函数相处呢有点小尴尬的,就是 gets() 在 scanf() 后边就爱捣乱。为什么呢,先了解它们两者之间的异同: 同: 都是可以接受连续的字符数据 并在字符结束后自动加上 '\0',标志结束接受 异: scanf 不能接受空格、制表符Tab、回车等,遇空格时就结束 ......

  

  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);
}