fgets()不是从文本文件中读取的?

时间:2013-11-24 07:31:56

标签: c null load settings gets

我有一个函数loadsets()(加载设置的缩写),它应该从名为 Progsets.txt 的文本文件加载设置。 loadsets()在成功时返回0,在检测到致命错误时返回-1。但是,实际 Progsets.txt 读取的代码部分(三个 fgets())似乎全部失败并返回空指针,因此除了一堆空值之外根本不加载任何东西。我的代码有问题吗?当我运行代码时,fp是一个有效的指针,我能够打开它进行读取。那有什么不对?

此代码用于使用cmd加载我的基本文本编辑器程序的默认文本颜色。

标题:

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

#define ARR_SIZE 100

struct FINSETS
{
    char color[ARR_SIZE + 1];
    char title[ARR_SIZE + 1];
    char maxchars[ARR_SIZE + 1];
} SETTINGS;

载入集():

int loadsets(int* pMAXCHARS) // load settings from a text file
{
    FILE *fp = fopen("C:\\Typify\\Settings (do not modify)\\Progsets.txt", "r");
    char *color = (char*) malloc(sizeof(char*) * ARR_SIZE);
    char *title = (char*) malloc(sizeof(char*) * ARR_SIZE);
    char *maxchars = (char*) malloc(sizeof(char*) * ARR_SIZE);
    char com1[ARR_SIZE + 1] = "color ";
    char com2[ARR_SIZE + 1] = "title ";
    int i = 0;
    int j = 0;
    int k = 0;
    int found = 0;

    while (k < ARR_SIZE + 1) // fill strings with '\0'
    {
        color[k] = title[k] = maxchars[k] = '\0';
        SETTINGS.color[k] = SETTINGS.maxchars[k] = SETTINGS.title[k] = '\0';
        k++;
    }

    if (!fp) // check for reading errors
    {
        fprintf(stderr, "Error: Unable to load settings. Make sure that Progsets.txt exists and has not been modified.\a\n\n");
        return -1; // fatal error
    }

    if (!size(fp)) // see if Progsets.txt is not a zero-byte file (it shouldn't be)
    {
        fprintf(stderr, "Error: Progsets.txt has been modified. Please copy the contents of Defsets.txt to Progsets.txt to manually reset to default settings.\a\n\n");

        free(color);
        free(title);
        free(maxchars);

        return -1; // fatal error
    }

    // PROBLEMATIC CODE:

    fgets(color, ARR_SIZE, fp);      // RETURNS NULL (INSTEAD OF READING FROM THE FILE)
    fgets(title, ARR_SIZE, fp);      // RETURNS NULL (INSTEAD OF READING FROM THE FILE)
    fgets(maxchars, ARR_SIZE, fp);   // RETURNS NULL (INSTEAD OF READING FROM THE FILE)

    // END OF PROBLEMATIC CODE:

    system(strcat(com1, SETTINGS.color)); // set color of cmd
    system(strcat(com2, SETTINGS.title)); // set title of cmd
    *pMAXCHARS = atoi(SETTINGS.maxchars);

    // cleanup

    fclose(fp);
    free(color);
    free(title);
    free(maxchars);

    return 0; // success
}

Progsets.txt:

COLOR=$0a;
TITLE=$Typify!;
MAXCHARS=$10000;

编辑:以下是size()函数的定义。由于我只使用ASCII文本文件,我假设每个字符都是一个字节,并且可以通过计算字符数来计算文件大小(以字节为单位)。有什么可疑的吗?

尺寸():

int size(FILE* fp)
{
    int size = 0;
    int c;

    while ((c = fgetc(fp)) != EOF)
    {
        size++;
    }

    return size;
}

1 个答案:

答案 0 :(得分:0)

问题在于您使用size()功能。它重复调用文件句柄上的fgetc(),直到它到达文件末尾,递增一个值来跟踪文件中的字节数。

这不是一个错误的方法(虽然我确定有更好的方法不涉及低效的基于字符的I / O)但它确实有一个致命的你似乎忽视的缺陷。

在你调用之后,你已经将文件一直读到最后,以便进一步读取,例如:

fgets(color, ARR_SIZE, fp);
由于你已经在文件的末尾,

将会失败。在从rewind()返回之前,您可能需要考虑size()之类的内容 - 这会将文件指针放回文件的开头,以便您可以再次阅读它。