【关于PTA平台中出现的问题】warning: ignoring return value of ‘scanf’, declared with attribute warn_unused_result

结论:该warning并不影响程序执行,而出现测试用例未通过的原因是程序本身存在其他错误,诸如逻辑问题,请仔细检查程序

1.问题复现

  • 首先仔细读这段英文警告:意思是,scanf的返回值正在被忽略,在warn_unused_result中有说明。

直接定位问题:你的程序忽略了scanf的值,这可能会产生一些问题,所以GCC编译器给了你一个警告。

如下图代码所示,在PTA平台上进行代码测试,出现warning测试用例部分通过或者全部失败。
【关于PTA平台中出现的问题】warning: ignoring return value of ‘scanf’, declared with attribute warn_unused_result_第1张图片

2.问题溯源

需要明确

  • warning并不是导致测试用例失败的原因,如果测试用例失败,请查找其他原因,如格式错误、逻辑错误,以及执行中出现的问题
  • warning一般在本地编译器内不会出现
  • PTA平台的编译器使用的是GCC编译器,该编译器会对scanf的返回值进行检查,如果忽略scanf的返回值,将会出现警告。
  • 通过查找相应资料,发现根本原因在以下几个方面:

(1)scanf返回值

  • scanfC标准库stdio里面定义的用于获取用户输入的函数,具体的介绍可以在CppReference上看到。
  • scanf的返回值是已经成功赋值的变量个数
  • 也就是说在使用scanf("%d %d",&a,&b)
    这个调用返回的值:
    • 用户输入1 2时返回的是2
    • 用户输入的是1 b,那么返回值为1
    • 如果输入的是a b,那么返回的是0
    • 如果遇到错误或遇到end of file,返回值为EOF
      说明格式控制符跟scanf的返回值有关

(2)scanf返回值可能造成的问题

  • 先测试如下代码:
#include "stdio.h"
int main(void)
{
  int x = 0;
  while(1)
  {
    x = scanf("%d", &x);
    printf("%d", x);
  }
  return 0;
}
  • 当我们一不小心输入了一个字符,而非数字,后果会是怎样的呢?你可以自己测试以下。
  • 结果:输入字母,屏幕上不间断输出0
  • 原因分析

scanf这里要求输入参数为数字,输入字母会导致赋值失败,因此scanf返回值一直为0,也就导致输出0。

为什么一直输出0?
是因为scanf对于接受失败的数据``会送回stdin缓冲区,之前输入的字母会一直存在。在这个死循环中,scanf一遍遍读缓冲区的字母,也就会一直输出已一大串0。要想像输入数字那样可以循环输入,必须清除掉缓冲区的数据:setbuf(stdin,NULL);这个语句加到scanf后面,如下所示:

  • 代码改进:
#include "stdio.h"
int main(void)
{
  int x = 0;
  while(1)
  {
    x = scanf("%d", &x);
    setbuf(stdin, NULL);
    printf("%d", x);
  }
  return 0;
}
  • 再来看下面这个例子,来自Stack Overflow网站中的提问
#include "stdio.h"
int main()
{
  int t;
  scanf("%d", &t);
  printf("%d", t);
  return 0;
}

这里面提到了GCC中关于warn_unused_result的相关文档表明:

在GCC编译平台中,对scanf的返回值进行检查。如果没有考虑对scnaf的返回值进行检查,则出现警告

接下来将代码进行改进:

#include "stdio.h"
int main()
{
  int t;
  if (scanf("%d", &t) == 1)
  {
    printf("%d", t);
  }
  else
  {
    printf("Failed t oread integer.\n");
  }
  return 0;
}
  • 可以看出,上述代码是用ifscanf的返回值进行判断,如果不满足条件则拒绝执行if中的代码。这样就scanf的警告即可消除。

3.解决办法

用条件判断语句对scanf的内容进行判断保护,避免输入非法字符,导致返回值为0或者出错。
例如:

#include "stdio.h"
int main()
{
  int x;
  if(scanf("%d", &x))
  {
  //在判断输入有效的情况下,执行本逻辑块代码
  }
}

warning不会对程序本身逻辑造成影响,忽略即可。程序未通过测试,要检查代码本身的错误。

你可能感兴趣的:(C语言学习笔记)