C语言 从二进制文件中读取 fread() 的意外行为



该程序基本上创建了一个包含所有信息的航班列表(由 ListaVoli.bin 的 fread() 函数读取)。该列表主要由节点组成,每个节点包含一个外部测试版。

#include <stdio.h>
#include <stdlib.h>
#define FILE_NAME "/Users/Matt/Downloads/ListaVoli.bin"
struct flight {
    int flightCode;
    char destination[3];
    int scheduledDepHour;
    int scheduledDepMinute;
    int realDepHour;
    int realDepMinute;
    int passengers;
};
struct node {
    struct flight volo;
    struct node *next;
};
struct node *addToList(struct node *list, struct flight voloIn) {
    struct node *newNode;
    newNode = malloc(sizeof(struct node));
    if (newNode == NULL) {
        printf("Error: malloc failedn");
        exit(EXIT_FAILURE);
    }
    newNode -> volo = voloIn;
    newNode -> next = list;
    return newNode;
}
void printList(struct node *list) {
    for (; list != NULL; list = list -> next) {
        printf("Code:%dnDestination:%snDeparture:%d-%dnReal:%d-%dnPassengers:%dnnn",
        list -> volo.flightCode,
        list -> volo.destination,
        list -> volo.scheduledDepHour,
        list -> volo.scheduledDepMinute,
        list -> volo.realDepHour,
        list -> volo.realDepMinute,
        list -> volo.passengers
        );
    }
}
void decolla(struct node *list, int flightCode, int realDepHour, int realDepMinute) {
    for (; list != NULL; list = list -> next) {
        if (flightCode == (list -> volo.flightCode)) { /*
            printf("Inserisci ora di partenza per il volo %d: ", flightCode);
            scanf("%d", &(list -> volo.realDepHour));
            printf("Inserisci minuto di partenza: ");
            scanf("%d", &(list -> volo.realDepMinute)); */
            list -> volo.realDepHour = realDepHour;
            list -> volo.realDepMinute = realDepMinute;
        }
    }
}
void delay(struct node *list) {
    for (; list != NULL; list = list -> next) {
        if ((list -> volo.realDepHour) - (list -> volo.scheduledDepHour) == 0) {
            if ((list -> volo.realDepMinute) - (list -> volo.scheduledDepMinute) > 5 && (list -> volo.realDepMinute) - (list -> volo.scheduledDepMinute) < 30) {
            printf("Il volo %d ha più di 5 minuti di ritardon", list -> volo.flightCode);
            continue;
            }
            if ((list -> volo.realDepMinute) - (list -> volo.scheduledDepMinute) > 30) {
                printf("Il volo %d ha più di 30 minuti di ritardon", list -> volo.flightCode);
                continue;
            }
        } else
            printf("Il volo %d ha più di 30 minuti di ritardon", list -> volo.flightCode);
    }
}
void passengersCount(struct node *list) {
    for (; list != NULL; list = list -> next) {
        if (list -> volo.passengers > 200) {
            printf("Il volo %d ha più di 200 passeggerin", list -> volo.flightCode);
            continue;
        }
    }
}
int main() {
    FILE *fp;
    struct node *first = NULL;
    struct flight volo;
    /* Apro il file e controllo che sia stato aperto correttamente */
    if ((fp = fopen(FILE_NAME, "rb")) == NULL) {
        printf("Can't open %sn", FILE_NAME);
        exit(EXIT_FAILURE);
    }
    for (int i = 0; i < 4; i++) {
        fread(&volo, sizeof(int), 7, fp);
        first = addToList(first, volo);
    }
    decolla(first, 3497, 11, 30);
    decolla(first, 2193, 11, 53);
    decolla(first, 4284, 11, 07);
    decolla(first, 5536, 12, 26);
    printList(first);
    delay(first);
    passengersCount(first);
    /* Controllo che il file sia chiuso correttamente */
    if (fclose(fp) == EOF) {
        printf("File not closed properly!");
        exit(EXIT_FAILURE);
    }
    return 0;
}
代码

编译正确,所以不用担心整个代码,专注于main()函数和两个结构。关于main()函数中的fread()函数,我有两个问题:

  1. 为什么,如果我把 sizeof(int) 作为第二个参数,航班目的地值被正确分配?char[3] 变量不应该大于 sizeof(int) 吗?
  2. 为什么,如果我把 sizeof(结构飞行)作为第二个参数(这将是最好的选择),我得到分段错误:11?

三个字符的数组是三个字节。int通常是(至少在现代 32 位和 64 位平台上)4 个字节。它可以读取sizeof(int),因为编译器添加了填充

但是读取结构

的"正确"(或至少通常)方法是将整个结构作为一个单元读取,即在您的情况下使用 sizeof(volo)

fread(&volo, sizeof(volo), 1, fp);

如果您因此而遇到其他错误,那么您正在做其他错误

相关内容

  • 没有找到相关文章

最新更新