检查C中的stdin管道是否为空

时间:2017-01-26 03:32:22

标签: c stdin

我正在使用用户输入来填充二维数组。用户在一行中输入数字然后我使用嵌套循环来填充数组:

//User inputs: "1 2 3 4 5"

for(i = 0; i < r; i++){
    for(j = 0; j < c; j++){
            scanf("%d", &arr[i][j]);
    }
 }

然而,问题是如果用户在有6个空间时输入5个整数,它只是等待另一个输入。如何检测数字是否不足?

我尝试过使用它,但它不起作用:

for(i = 0; i < r; i++){
    for(j = 0; j < c; j++){
        if (!feof(stdin)){
            scanf("%d", &arr[i][j]);
        }
        else{
            printf("insufficient datapoints\n");
        }
    }
 }

2 个答案:

答案 0 :(得分:1)

您可以在流中使用peek并在实际使用它们之前测试字符。 (等等于c)。

您可以使用它来忽略空格(您需要这样做)。

您还可以使用此隐藏值来指示是否输入了不足的字符。

需要在实际读取(scanf)之前完成窥视。

下面添加了粗略的示例代码
#include <stdio.h>
#include <ctype.h>

int r=3; 
int c=2;

int arr[100][100]; // FIX magic

int main(int argc, char** argv[]) {

  for(int i=0; i<r; i++) {
    for(int j=0; j<c; j++) {

      if (feof(stdin)) {
        // error check and error / normal exit etc.
        printf("eof\n");
      }

      char c=getchar();

      if (c=='\n') {
        // error check and error / normal exit here
        printf("newline\n");

      } else if (isspace(c)) {
        // advance and remove them - watch for end of stream when winding
        printf("advance and discard whiitespace\n");

      } else { // add check for isdigit
        // push back
        ungetc(c, stdin);
        printf("ungetc\n");
      }

      scanf("%d", &arr[i][j]);  
      printf("got %d\n", arr[i][j]);   
    }    
  }

  return 0;
}

答案 1 :(得分:1)

实现目标的一种方法是使用fgets()代替scanf()来一次读取一行输入。然后strtok()可用于将输入行分解为标记,strtol()可用于将标记解析为数字。与scanf()相比,使用fgets来处理非结构化用户输入要容易得多。

以下代码执行此操作。如果输入行上的元素太多,元素太少,或者其中一个元素不是有效数字,则会打印一条消息,并且必须再次输入该行。

当用户输入每一行时,strtok()用于将行划分为令牌。令牌分隔符列表存储在delims[]中。请注意,令牌可以用空格或制表符分隔;分隔符本身不是令牌的一部分,因此包括\r\n可确保这些字符不会成为行中最终标记的一部分。

找到令牌时,strtol()用于将其转换为整数(如果可能)。在调用strtol()之后,指针tail指向令牌中不属于数字的第一个字符;如果tail指向NUL终止符,则整个字符串被解析为数字,否则输入被视为坏,并且必须再次输入该行。

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

#define BUF_SIZE  1000

int main(void)
{
    size_t r = 3;
    size_t c = 5;
    size_t i, j;
    char buffer[BUF_SIZE];
    char *token;
    char *tail;
    const char delims[] = " \t\r\n";
    int arr[r][c];
    int temp_val;

    printf("Enter rows of %zu data elements:\n", c);

    for(i = 0; i < r; i++){
        j = 0;

        if (fgets(buffer, BUF_SIZE, stdin) == NULL) {
            perror("Error in fgets()");
            exit(EXIT_FAILURE);
        }

        token = strtok(buffer, delims);
        while (token != NULL) {
            temp_val = strtol(token, &tail, 10);
            if (*tail == '\0') {
                arr[i][j] = temp_val;
                ++j;
            } else {                // token not a valid number
                j = 0;
                break;
            }

            if (j > c) {            // too many input values
                break;
            }

            token = strtok(NULL, delims);
        }

        if (j != c) {
            printf("insufficient datapoints\n");
            --i;                    // enter row again
        }
    }

    for (i = 0; i < r; i++) {
        for (j = 0; j < c; j++) {
            printf("%5d", arr[i][j]);
        }
        putchar('\n');
    }

    return 0;
}

示例互动:

Enter rows of 5 data elements:
1 2 3 4
insufficient datapoints
1 2 3 4 5 6
insufficient datapoints
1 x 2 3 4
insufficient datapoints
1 2 3 4 x
insufficient datapoints
1 2 3 4 5 x
insufficient datapoints
1 2x 3 4 5
insufficient datapoints
1 2 3 4 5
2 3 4 5 6
 3  4  5    6 7
    1    2    3    4    5
    2    3    4    5    6
    3    4    5    6    7