我正在使用用户输入来填充二维数组。用户在一行中输入数字然后我使用嵌套循环来填充数组:
//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");
}
}
}
答案 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