【问题标题】:What is the best way to read a string?读取字符串的最佳方法是什么?
【发布时间】:2014-05-10 14:26:09
【问题描述】:

所以我真的很困惑。我必须用 C 编写一个程序,它基本上是一个通讯录,因此我必须从用户那里获取多个字符串(姓名、ID、电话等)

一开始我尝试只使用scanf(),但有时它会与换行符'\n' 混淆。经过一番谷歌搜索,我最终使用scanf() 获取单个字符或整数(用户回答是或否问题,或从菜单中选择一个操作)和fgets() 读取地址簿的字段。但是,在使用scanf() 之后,我还必须多次使用fflush(stdin),正如我所见,这里不建议这样做。此方法按预期工作。

那么从用户那里读取字符串的最佳方式是什么? fflush(stdin) 不提供便携性吗?这是一项任务,所以我也必须考虑可移植性,因为我将在另一台计算机上执行我的代码。

提前谢谢你。

编辑:这就是我到目前为止所得到的。请原谅一些用另一种语言(阿尔巴尼亚语)写的词。我相信你能理解这是怎么回事。

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

void regjistrim();
void kerkim();
void modifikim();
void fshirje();
void rradhitje();
void display();

#define EMRI 50
#define MBIEMRI 50
#define ID 20
#define TEL 20
#define EMAIL 25

typedef struct addressbook
{
    char emri[EMRI];
    char mbiemri[MBIEMRI];
    char id[ID];
    char tel[TEL];
    char email[EMAIL];
} addressbook;

FILE* Addressbook;

int main(void)
{
    char input[2];
    int choice;

    printf("----------------ADDRESS BOOK----------------");
    printf("\n\n\t1 - Regjistrimi i ri\n");
    printf("\n\t2 - Kerkim\n");
    printf("\n\t3 - Modifikim\n");
    printf("\n\t4 - Fshirje\n");
    printf("\n\t5 - Rradhitje\n");
    printf("\n\t6 - Afishim i address book\n");
    printf("\n\t0 - Exit\n");

    fgets(input, 2, stdin);
    sscanf(input, "%d", &choice);

    while (choice < 0 || choice > 6)
    {
        printf("\nShtypni nje numer nga 0 - 6: \n");
        fgets(input, 2, stdin);
        sscanf(input, "%d", &choice);
    }

    switch (choice)
    {
        case 1:
            regjistrim();
            break;
        case 2:
            kerkim();
            break;
        case 3:
            modifikim();
            break;
        case 4:
            fshirje();
            break;
        case 5:
            rradhitje();
            break;
        case 6:
            display();
            break;
        case 0:
            exit(0);
            break;
    }

    return 0;
}


//Regjistron nje qytetar ne addressbook
void regjistrim()
{
    char answer;

    addressbook entry;
    do
    {
        Addressbook = fopen("Addressbook.txt", "a+");

        printf("\nShtypni emrin: ");
        fgets(entry.emri, EMRI, stdin);

        printf("\nShtypni mbiemrin: ");
        fgets(entry.mbiemri, MBIEMRI, stdin);

        printf("\nShtypni ID-in: ");
        fgets(entry.id, ID, stdin);

        printf("\nShtypni nr. telefoni: ");
        fgets(entry.tel, TEL, stdin);

        printf("\nShtypni email-in: ");
        fgets(entry.email, EMAIL, stdin);

        fprintf(Addressbook, "Emri: %sMbiemri: %sID: %sNr. telefoni: %sEmail: %s\n", entry.emri, entry.mbiemri, entry.id, entry.tel,entry.email);

        fclose(Addressbook);

        printf("\nShtypni y/Y neqoftese doni te regjistroni person tjeter: ");
        fgets(answer, 1, stdin);
    }
    while(answer == 'y' || answer == 'Y');
}

【问题讨论】:

  • 并非不推荐fflush(stdin)。在这种情况下,刷新输入流是 error (它会导致未定义的行为)。只需使用fgets() 独占,然后解析字符串。不要尝试将解析和用户输入阶段结合起来。
  • 我支持user3477950的观点。我想补充一点,一般来说scanf 不被认为是读取无法保证结构的输入的好方法。这是非常危险的,并且很难或根本不可能安全地读取输入。我不确定是否有任何 best 方法。如果你想正确地做到这一点,你最终会得到一些库,或者将所有内容作为字符串读取并稍后解析缓冲区。
  • 抱歉,我似乎无法理解解析是什么。另外,我将如何使用fgets() 从用户那里获取一个 int?
  • @user3484582 在这种情况下,“解析”意味着弄清楚输入的内容。伪代码应为 1。获取输入行。 2. 判断该行是否包含整数。
  • @user3484582 我看到的一个问题是 fgets 的输入缓冲区应该足够大以容纳整行(例如,所有字符加上一个 '\n' 加上一个 '\0')。不建议使输入缓冲区仅包含 2 个字符。我会制作一个大约 10000 个字符或其他内容的缓冲区。当你得到一行时,将相关字符复制到你的结构中

标签: c string


【解决方案1】:

使用scanf,您可以像这样清除换行符。另外,我也加入了fgets

#include<stdio.h>

int main(void)
{
    int i, N = 5;
    char buffer[N];

    printf("Enter %d characters\n", N+1);

    scanf("%5s", buffer);   /* MUST check comments below on this page for this! */

    /* Clear trailing input */
    while(getchar() != '\n')
        /* discard */ ;


    for(i = 0 ; i < 5 ; ++i)
        printf("|%c|\n", buffer[i]);

    printf("End with scanf\n\n");

    /*****************************************************/

    printf("Enter %d characters\n", N+1);

    fgets(buffer, 5, stdin);

    for(i = 0 ; i < 5 ; ++i)
        printf("|%c|\n", buffer[i]);

    printf("End with fgets\n\n");

    return 0;
}

此外,此代码演示了您可以对输入的每个函数设置的限制。

Source

【讨论】:

  • 效果很好,但它似乎是一个草图解决方案。可以用getch()吗?
  • @user3484582 吃掉尾随的换行符?是的。
  • @user3484582,如果你对答案没问题,请告诉我,这样我就可以注销而不用担心了。 :)
  • 是的,正如我之前所说,它工作得很好,但现在我正试图找出一种只使用fgets() 的方法。感谢您的帮助。
  • @user3484582 我不知道接受这个答案是否会更好(因为它对你来说没问题)并为你拥有的程序开始一个新问题,
【解决方案2】:

好吧,“最佳”可能有点主观,但我发现 做一个单独的函数来读取数字会使事情发生 如果你不是绝对需要它,那就更容易一点,避免 scanf,例如

int readNumber(int min, int max)
{
  char number[32];
  do
  {
    if ( fgets( number, sizeof(number), stdin ) != NULL )
    {
      // note that if 'number' is not a number atoi returns 0
      int n = atoi(number); 

      if ( n>= min && n <= max ) 
      { 
        return n;
      }
    }
    printf( "please enter a valid value between %d and %d\n", min, max );
  }
  while ( 1 );

  return -1; // never reached
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2011-10-03
    • 1970-01-01
    • 1970-01-01
    • 2016-01-07
    • 2021-11-16
    • 2012-07-28
    • 1970-01-01
    相关资源
    最近更新 更多