0

对于我的一生,我无法与 c 字符串和输入/输出和平相处。

对于我的程序,我只需输入一个字符串,然后在以下代码中对其进行处理:(tmpstring 和 ch 已经定义)

对于我的输入,我在终端中写:echo "test" | 。/程序

    int main(int argc, char *argv[]) {
    char tmpstring[2048];
    int ch;
    int r;
    int c;

    fgets(tmpstring, sizeof tmpstring, stdin);
    while((ch = fgetc(stdin))!= EOF && ch != '\n');
    tmpstring[strlen(tmpstring)-1]='\0';

    strncpy(opponent, tmpstring+1, strlen(tmpstring+1));

    move();

move() 内部;

    char buffer[2048]={0};
    int r, c;
    r=0; c=0;
    printf("Your move (row column):");
    if((fgets(buffer, sizeof buffer, stdin)==NULL) || ((sscanf(buffer,"%d %d", &r, &c))!=2)){
            printf("Invalid input. Please insert two numbers separated by whitespace.\n");
            exit(1);
    }
    //continues

执行此操作直接进入无效输入,而不要求更多输入。我已经阅读了所有关于你不应该清除标准输入的内容(这是不可能的),但我真的不知道该怎么做。我显然试图用 while 循环的第二部分“转储”标准输入。我已将第一个条件后的 && 更改为 || 在while循环中。不用找了。总的来说,在已经使用过 fgets 之后,我怎么能要求更多的输入呢?

*编辑:更多代码并分离原始的while循环

4

1 回答 1

0

这类问题已在别处讨论过。
我将从这里复制我自己的帖子的一部分:

使用 scanf -> ANSI C 的最大字符串长度

  • 它通过调用库定义了一个my_scanf()具有可变数量参数的函数,并与 和结合。 stdarg.hfgets()vsscanf()
  • 该函数旨在以正确的方式处理fgets()和的组合sscanf()(实际上,我们必须使用vsscanf()它才能正确处理参数列表)。
  • 输入被读取到字符的上限。如果输入超过此限制,则将其截断。'\n' 被正确处理。该函数返回的值与返回的值相同scanf()。所以,你可以my_scanf()改用。

这里有代码:

#include <stdio.h>
#include <stdarg.h>

int my_scanf(const char* fmt, const unsigned int maxbuff, ...) {
    va_list ptr;
    int ret;

    if (maxbuff <= 0)
       return EOF; /* Bad size for buffer[] */

    char buffer[maxbuff+1];
    buffer[maxbuff-1] = '\0';  /* Quick buffer cleaning... */

    if (fgets(buffer, maxbuff+1, stdin) == NULL)
       return EOF; /* Error detected */
    else {
        if ((buffer[maxbuff-1] != '\n') && (buffer[maxbuff-1] != '\0'))
            /* Condition logically equivalent to:
                   fgets() has not reached an '\n'
            */
            while (getchar() != '\n')
               ; /* "Flushing" stdin... */

        va_start(ptr, maxbuff);
        ret = vsscanf(buffer, fmt, ptr);
        va_end(ptr);
        return ret;
    }    
}

#define MAXBUFF 20
int main(void) {
   int x; 
   float z;
   int scanf_ret = my_scanf("%d %g", MAXBUFF, &x, &z);
   printf("\nTest:\n x == %d\n z == %g\n scanfret == %d", x, z, scanf_ret);
   getchar();   
   return 0;   
}

函数 my_scanf() 具有原型

int my_scanf(const char* fmt, const int maxbuff, ...);
  • 它接受一个格式字符串fmt,其行为方式与任何其他scanf()-like 相同。
  • 第二个参数是从标准输入(键盘)有效接受 的最大字符数。
  • 返回值是一个intEOF如果maxbuff没有意义,或者发生了一些输入错误。如果返回非负值,则与标准函数sscanf()vsscanf().

    1. 在函数内部,maxbuff以 1 递增,因为fgets()为额外的 '\0' 字符腾出了一些空间。
    2. 的非正值maxbuff立即被丢弃。
    3. fgets()将从stdin(keyboard) 读取一个字符串,其中最多可容纳maxbuff字符,包括 '\n'。
    4. 如果用户输入了一个很长的字符串,那么它将被截断,并且需要某种“刷新”机制才能将所有字符丢弃到下一个 '\n' ( ENTER )。如果不是,则下一个键盘读数可能包含较旧的字符,根本不需要。
      • "flushing" 的条件是 readfgets()后还没有达到 '\n' stdin
      • buffer[maxbuff - 1]当且仅当不等于 '\0' 或 '\n'时,这是真的。(检查它!
    5. Finally, an appropiate (and standard) combination of stdarg.h macros and the function vsscanf() is employed to process the variable list of parameters.
于 2013-08-27T05:31:47.077 回答