Как удалить записи из относительного двоичного файла? - PullRequest
0 голосов
/ 24 марта 2019

Я имею в виду ручное удаление записей с оценками [5] ниже 5. Я еще ничего не пробовал, потому что я не знаю, с чего начать. Я не хочу, чтобы функция выполняла работу быстро, Я хочу реализовать функцию и понять ее, но я не знаю, как удалять записи в относительных файлах. Мой код должен работать следующим образом: - двоичный файл должен получать данные из текстового файла и показывать искомого учащегося, если учащегося не существует, я должен добавить его. После всего этого я смогу удалить учеников со степенью [5] строго ниже 5. В моем коде отсутствует только функция удаления.

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <conio.h>

typedef struct {
    int nrm;             //enrolment number - works as relative key
    char CNP[14];        //string ASCIIZ
    char nume[30];       //string
    int an;              //study year
    int grupa;           //group
    unsigned char note[20];  //grades
    char is;//this is status indicator
} STUDENT;

int filesize(FILE* f, int rec_size)
{//from the crt.pos substract one and i ll know the record of...
    long crt_pos;
    int size;
    //below compute the size of the file in records, withouth altering the crt.pozition
    crt_pos = ftell(f);//save the current pozition in a var, before going to the end of the file
    fseek(f, 0, SEEK_END);//go to the end of the file
    size = ftell(f) / rec_size;//ftell gives the current position and the dim of the record
    fseek(f, crt_pos, SEEK_SET);//ne readuce la positia curenta, initiala pe care am salvat o n var
    return size;
}

void main()
{
    char numefr[30] = "..\\Studenti_r_f.dat";//why \\? changes the meaning of the foll char; afiseaza un singur
    FILE* f;
    STUDENT x;//record variable
    int i, key, dim;//key=key for the record;dim=size of file in records
    //dim changes when computing the sizeof(file), at every iteration
    fopen_s(&f, numefr, "wb+");
    //optionally preform the file

    freopen("..\\input.txt", "rt", stdin);  //remove if reading data from keyboard and uncomment the printf lines

    //printf("Enrollment number: ");
    scanf_s("%d", &key);//read the first field, always the 1 field is the key
    while (!feof(stdin))//process the key read another and so on
    {
        //check key range
        dim = filesize(f, sizeof(STUDENT));//dim=nr de studenti din fisier;nr de recorduri;
        //check the poz indicated by the key, it may have changed since the last iteration
        if (key >= dim)//if the key=10, position 10 doesn t exist
        { //extend the file
            x.is = 0;
            fseek(f, 0, SEEK_END);//go to the end of file and write this record
            for (i = 0; i < key - dim; i++)
                fwrite(&x, sizeof(STUDENT), 1, f);
        }
        fseek(f, key * sizeof(STUDENT), SEEK_SET);//change the pos in the file, go to the pos indicated by the key
        fread(&x, sizeof(STUDENT), 1, f);//read rec from file
        if (x.is == 1)//if is=1, the position is taken, it must always be 0
            printf("\nError: duplicate key %d. Skipping.", key);//read rest of field for student, write record from file then
        else
        {
            x.nrm = key;
            //printf_s("Name: ");
            gets_s(x.nume);
            gets_s(x.nume);
            //printf_s("CNP: ");
            gets_s(x.CNP);
            //printf_s("Year: "); 
            scanf_s("%d", &x.an);
            //printf_s("Group: ");
            scanf_s("%d", &x.grupa);
            for (i = 0; i < 20; i++)
                //x.note[i]=0;
                scanf_s("%d", &x.note[i]);
            x.is = 1;
            //the cur poz is after this record before we write the fseek, which brings us to the beginning of this record
            fseek(f, key * sizeof(STUDENT), SEEK_SET);//ne aduce la inceputul recordului nou adaugat;
            fwrite(&x, sizeof(STUDENT), 1, f);//write a record
        }//everything repets with a new key
        //printf_s("Enrollment number (or CTRL+Z): ");
        scanf_s("%d", &key);
    }

    fclose(f);
    fclose(stdin);

    printf("\nDone. File <%s> was created. Press a key.", numefr);
    _getch();
}

1 Ответ

1 голос
/ 24 марта 2019

Есть два способа удалить запись из файла:

1) скопировать в новый файл все записи, предшествующие удаляемому, пропустить удаляемую запись, затем скопировать все остальные записи, затем удалить исходный файл, затем переименовать новый файл в исходное имя файла .

2) пошагово просматривайте файл, пока не найдете удаляемую запись. Затем в цикле скопируйте следующую запись в текущую запись, затем обрежьте файл на 1 длину записи

Все это (выше) может быть очень грязным и трудоемким. Предложить: просто установите поле 'is' в записи, которое будет 'удалено', на 0

...