【问题标题】:fgets not waiting for user inputfgets 不等待用户输入
【发布时间】:2016-05-06 00:23:50
【问题描述】:

在给定的代码中,fgets 不等待输入。

我尝试使用 scanf,但它给出了异常错误(在 0x0F74DDF4 (ucrtbased.dll) 处引发异常)。我正在使用 Visual Studio 2015 来调试我的代码。谁能解释为什么 fgets 不等待输入?

#include<stdio.h>
#include<stdlib.h>
#include<process.h>

//GLOBAL-VARIABLE DECLARTION
#define MAX 1000

//GLOBAL-STRUCTURES DECLARATION
struct census
{
char city[MAX];
long int p;
float l;
};

//GLOBAL-STRUCTURE-VARIABLE DECLARATION
struct census cen[] = { 0,0,0 };

//USER-DEFINED FUNCTION
void header();

void header()
{
printf("*-*-*-*-*CENSUS_INFO*-*-*-*-*");
printf("\n\n");
}

//PROGRAM STARTS HERE
main()
{   
//VARIABLE-DECLARATION
int i = 0, j = 0;
//int no_of_records = 0;

//FUNCTION CALL-OUT
header();

printf("Enter No. of City : ");
scanf_s("%d", &j);

printf("\n\n");

printf("Enter Name of City, Population and Literacy level");
printf("\n\n");

for (i = 0;i < j;i++)
{
    printf("City No. %d - Info :", i + 1);
    printf("\n\n");

    printf("City Name :");
    fgets(cen[i].city, MAX, stdin);
    printf("\n");

    printf("Population : ");
    scanf_s("%d", &cen[i].p);
    printf("\n");

    printf("Literacy : ");
    scanf_s("%f", &cen[i].l);
    printf("\n");
}

//TERMINAL-PAUSE
system("pause");
}

【问题讨论】:

  • @user3121023 有什么办法解决这个问题吗?
  • @Szymson scanf 在 Visual Studio 中不起作用。我们必须使用 scanf_s 和它给我的错误。
  • 尝试使用 fgets() 到一个单独的 char 数组中,然后 sscanf() 从中解析。
  • 你不能只用scanf_s替换scanf;如果您有%s%c%[ 格式,当您想模拟fgets 时,您必须为scanf_s 中的每个格式提供最大缓冲区大小。但我支持@jamieguinan:在整个交互式输入中使用两步方法:首先读取一行,然后从该行读取数据。
  • scanf_s("%f", &amp;cen[i].l); 不使用数字后面的'\n'。然后下一个fgets(cen[i].city, MAX, stdin); 读取该1 个字符'\n'。干脆不要用scanf()

标签: c input fgets


【解决方案1】:

我总是使用fgets,后跟sscanf

在顶部声明,

char line[MAX];

然后用fgets得到一行,用sscanf从中解析出int值,

printf("Enter No. of City : ");
fgets(line, sizeof(line), stdin);
sscanf(line, "%d", &j);

l 的类似模式

printf("Literacy : ");
fgets(line, sizeof(line), stdin);
sscanf(line, "%f", &cen[i].l);

【讨论】:

  • 顺便说一句,如果你为 j 使用大于 3 的值,你会溢出你的 `cen]` 数组,但这是一个单独的问题。
  • 像@user694733 那样检查返回值会很好。我的示例是对原始示例的简单修改,它应该适用于健全的输入。
  • 你能告诉我为什么当我为 j 使用大于 3 的值时它会溢出吗?
  • @Eddy,当您在括号中声明 cen[] 时没有大小,编译器会将其大小调整为 {}s 中的项目数,其中有 3 个零。您可以声明它cen[10] 并处理最多 10 个,或者任意数量。为了安全起见,如果j 比您的数组大,您可以在循环之前放置和if 测试并中断/返回/退出。
  • 我明白了 jamie 但谢谢回复:)
【解决方案2】:

当您输入城市数量并按 Enter 时,scanf 不会从输入中读取换行符。 fgets 然后尝试读取但发现换行符,并立即停止。

不要用scanf读数字,先用fgets读入字符串,再用sscanf(或strtol/strtof/strtod)转成数字。

char temp[LIMIT];
if(fgets(temp, LIMIT, stdin)) {
    if(sscanf(temp, "%d", &j) == 1) {
        // Ok
    }
    else {
        // Handle sscanf error
    }
}
else {
    // Handle fgets error
}

【讨论】:

  • 谢谢你的帮助,我真的很感激 :)