将结构对象输出到二进制文件并从中输入回 C

Output struct objects to binary file and input from it back in C

我用C写管理系统。 我有这个结构

#define STRING_LENGTH 32
typedef struct {
    char name[STRING_LENGTH];
} Name;

typedef struct{
    int id, balance;
    Name clientName;
} Client;

我创建了几个测试对象,打开二进制文件进行写入,使用 fwrite 将对象写入文件,关闭它,然后使用 fread 进入 while(!feof... 块,然后我问题是我将 4 个对象打印到二进制文件,当我从文件中读取对象并将其打印到屏幕时,最后一个对象打印了两次。 我做错了什么?我只需要将对象写入文件,然后从中取回。

我的代码:

FILE *clientsFile = NULL;

switch (selectedOption)
{
case CLIENT:

        clientsFile = fopen(CLIENTS_FILE_PATH, "wb");

        Client getUser;
        Client temp1 = { 1, 10000, "Alex" };
        Client temp2 = { 2, 100000, "Valery" };
        Client temp3 = { 3, 105466, "Jack" };
        Client temp4 = { 4, 1069640, "Pam" };

        fwrite(&temp1, sizeof(Client), 1, clientsFile);
        fwrite(&temp2, sizeof(Client), 1, clientsFile);
        fwrite(&temp3, sizeof(Client), 1, clientsFile);
        fwrite(&temp4, sizeof(Client), 1, clientsFile);

        fclose(clientsFile);

        clientsFile = fopen(CLIENTS_FILE_PATH, "rb");

        do
        {               
            fread(&getUser, sizeof(Client), 1, clientsFile);
            printf("ID : %d, Name : %s, Balance : %d\n", getUser.id, getUser.clientName.name, getUser.balance);
        } while (!feof(clientsFile));

        break;

输出照片:screen of output

感谢回答

feof() returns 如果已引发文件结束指示符标志,则为真。成功调用 fread() 不会导致升旗。因此,在读取最后一条记录后,您再次迭代。

相反,检查 fread() 是否成功,以确定您是否已到达文件末尾或遇到其他错误。

    do
    {               
        if (fread(&getUser, sizeof(Client), 1, clientsFile) == 1) {
            printf("ID : %d, Name : %s, Balance : %d\n", getUser.id, getUser.clientName.name, getUser.balance);
        } else {
            /* do something */
        }
    } while (!feof(clientsFile));

因为你用的是do-while

do
{               
    fread(&getUser, sizeof(Client), 1, clientsFile);
    printf("ID : %d, Name : %s, Balance : %d\n", getUser.id, getUser.clientName.name, getUser.balance);
} while (!feof(clientsFile));

在这里,当您第 5 次阅读时,您会收到 EOF。但是 "getUser" 仍然有第 4 个客户端条目。所以你得到了两次最后的输出。

解决方法: 改成while循环。

while (fread(&getUser, sizeof(Client), 1, clientsFile) && !feof(clientsFile))
{
    printf("ID : %d, Name : %s, Balance : %d\n", getUser.id, getUser.clientName.name, getUser.balance);
}

我的方法就是这样。如果您读取了正确数量的记录,则很好,如果不正确,请退出。没必要涉及feof().

while(fread(&getUser, sizeof(Client), 1, clientsFile) == 1) {
    printf("ID : %d, Name : %s, Balance : %d\n", getUser.id, getUser.clientName.name, getUser.balance);
}