fgets存储来自文本文件的未知数据

时间:2018-07-21 21:04:40

标签: c file text fgets

我正在创建一个猜单词游戏,该游戏每行读取一个文本文件,直到找到一个随机单词并将其存储在字符串(word)中。然后,用户输入字母,直到显示出所存储单词的所有字母为止。

到目前为止,它运作良好,但是每次都是第一个读取的单词时,一些未知字符会存储在char word[20]的开头。

请注意,我使用C移动应用程序,并且使用我认为的clang 6.0编译器。那么错误是来自我的代码还是来自他们的应用程序? (我喜欢)。

这是完整的,更清晰的代码:

//guess the right word

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

#define TRUE 1
#define FALSE 0
#define NB_OF_WORDS 3 //number of words in text file

main() {
    char begin, word[20] = { 0 }, guessedletter;
    int num, rightletter, success;
    int i = 0;
    int show[20] = { 0 };//shown letters

    FILE *ressource = NULL;
    srand(time(NULL));

    if ((ressource = fopen("ressource.txt", "r")) == NULL) {
        fprintf(stderr, "Error ressource.txt");//open in read only mode
        exit(EXIT_FAILURE);
    }

    printf("Welcome to Word Guess, a word guessing contest.\n"
           "You have to find the letters of a word and guess what word it is.\n"
           "Begin? (y/n)\n");

    while ((begin = getchar()) == 'y') { //game loop
        /*reinitializations*/
        fseek(ressource, 3, SEEK_SET);//replaces rewind(ressource)
        success = FALSE;
        rightletter = 0;
        num = 0;
        num = rand() % NB_OF_WORDS; //random number between 0 and NB-1
        for (i = 0; i < 20; i++) {
            show[i] = FALSE;
            word[i] = 0;
        }
        i = 0;
        /*end of reinitializations*/

        while (i <= num) {//reads line by line until random word is stored
            if (fgets(word, 20, ressource) == NULL) {
                fprintf(stderr, "fgets did not work");
                exit(EXIT_FAILURE);
            }
            i++;
        }

        printf("%s", word);//here is just for testing if the read word is well read. Which isn't the case if num=0 

        for (i = 0; i < 20; i++)
            if (word[i] == '\n')
                word[i] = '\0';//adds zero character to show where the string ends
        while (!success) { //guessing loop
            printf("\nWrite a letter: ");
            scanf("%c", &guessedletter);
            printf("\n");

            for (i = 0; word[i] != '\0'; i++) { //compares entered letter to letter from string. If they match...
                if (word[i] == guessedletter) {
                    if (!show[i])
                        rightletter++;//avoids letters entered twice from increasing count
                    show[i] = TRUE;
                }
                if (show[i])
                    printf("%c", word[i]);//...a letter is revealed...
                else
                    printf("*");//else it stays hidden
            }
            if (rightletter == strlen(word))
                success=TRUE;//if all the right letters found (same number of letters found as number of letters in the words) you win
        }

        printf("\nCongratulations you have won!\nDo you want to replay? (y/n)");
        getchar();//clears newline character
    }

    fclose(ressource);
    return 0;
}

当num = 0时,我在打印的单词之前得到一个奇怪的符号,好像文本文件的第一个字符不应该出现在这里一样。

此外,在猜谜游戏中,如果要猜的单词(单词[20])是“令人讨厌的”,那它是“ ressource.txt”(num = 0)中的第一个单词。一旦我猜到所有字母,该字就会像在屏幕上那样打印:讨厌。列表中的任何其他单词都不会发生这种情况。

我是这个网站的新手,正在通过手机发帖...对您的任何错误表示歉意。

编辑:删除了fgets的fgetc。如果fgets读取第一行,仍然会得到几个未知字符。

编辑2:添加了整个代码,将mot [20]转换为word [20],添加了错误测试

编辑3:替换倒带(资源);通过fseek(ressource,3,SEEK_SET);解决了问题。这意味着在文本文件的开头确实存在三个未知字符

1 个答案:

答案 0 :(得分:2)

已发布的代码中存在多个问题:

  • 您仅发布了一段代码,该函数的其余部分可能会导致无法从您发布的内容中进行分析的问题。请把完整的代码发布到出现问题的最小程序中。
  • 您不会在while循环中测试文件结尾。如果您尝试跳过的行数超过文件中的行数,则此循环将无限期地运行。
  • 您不测试fgets()的返回值:如果偶然在上一个while循环中跳过了文件的全部内容,则fgets()将失败并返回{{1 }},使NULL处于不确定状态,从而在随后的mot中引起意外行为。

编辑:修改后的代码仍不检查printf的返回值。

编辑:感谢您发布完整的代码,但是以这种方式修改问题会使此答案和注释无关。

您的字典文件fgets()似乎以UTF-8编码的BOM (Byte Order Mark)开头。我猜它包含法语单词,包括一些带有重音字母的单词,例如reculéesencodées...。您的文本编辑器将其保存为UTF-8编码,并带有额外的代码点开始时,其他程序可以轻松确定此编码。问题在于您的程序无法处理UTF-8。假设从ressource.txt读取的每个字节代表一个字符。它可能会偶然地将某些单词与重音字母匹配,但很可能找不到它们。