Удаление записи: функция delcd()
Функция, удаляющая информацию о диске, использует стандартный алгоритм для работы с последовательными файлами. Используя последовательный доступ, мы не можем непосредственно перейти к интересующему нас месту в файле, чтобы изменить содержащуюся там запись. Если открыть файл с режимом доступа "w", его содержимое будет уничтожено. При использовании режима доступа "a" мы сможем только добавить данные в конец файла.
Решение этой проблемы лежит в использовании двух файлов. Функция delcd() открывает файл с данными, используя режим доступа "r", и временный файл с режимом доступа "w":
if (fp = fopen(FILENAME, "r")) == NULL) { printf("Невозможно открыть файл %s\n", FILENAME); exit(); }if ((tp = fopen(tempfile, "w")) == NULL) { printf("Невозможно открыть файл %s\n", tempfile); fclose(fp); exit(); }После этого мы вводим название диска, который следует удалить:
printf("Введите название диска: ");gets(delname);Цикл while осуществляет чтение каждой структуры (записи) из исходного файла:
Если запись не является той, которую пользователь хочет удалить, функция delcd() заносит ее во временный файл:
if (strcmp(disc.name, delname) != 0)fwrite(&disc, sizeof(disc), 1, tp);В нашей программе функция strcmp() используется в нескольких местах для определения того, прочитана ли из файла нужная запись. Функция strcmp() не входит в K&R-стандарт Си, но она есть во всех библиотеках ANSI Си и Си++.
Встретив нужную запись, функция delcd() не копирует ее во временный файл, а вместо этого устанавливает флаг, чтобы впоследствии узнать, была ли структура удалена:
else fflag = 'y'; }Цикл while продолжает копирование из одного файла в другой до тех пор, пока не достигнет конца файла, после чего оба файла закрываются:
fclose(fp);fclose(tp);Если функция delcd() не нашла записи, которую пользователь пожелал удалить, она выведет на экран соответствующее сообщение, после чего завершит работу:
puts("\nДиск с таким названием не найден");pause = getchar();Произведя удаление записи, вы получите два разных файла. Исходный файл содержит все данные, включая и информацию, которую вы хотите удалить. Временный файл содержит только данные, которые требовалось сохранить, так как функция delcd() не записала в него структуру, подлежащую удалению. Но мы не можем оставить данные в таком виде, программа способна вносить дополнительные данные и выполнять другие функции лишь с исходным файлом, поэтому мы должны привести его содержимое в соответствие с полученным результатом.
Некоторые компиляторы Си++ поставляются с библиотечными функциями, позволяющими измененять имена файлов, но в некоторых компиляторах они отсутствуют. В общем виде мы должны снова открыть оба файла, теперь уже с обратным порядком доступа, то есть исходный файл открывается с режимом доступа "w", после чего данные из него будут удалены, а временный файл открывается с режимом доступа "r". Для этого используется функция openwr(), вызываемая функцией delcd() и другими функциями:
if (fp = fopen(FILENAME, "w")) == NULL) { printf("Невозможно открыть файл %s\n", FILENAME); exit(); }if ((tp = fopen(TEMPFILE, "r")) == NULL) { printf("Невозможно открыть файл %s\n", tempfile); fclose(fp); exit(); }
После того как файлы открыты, функция delcd() считывает записи из временного файла и переносит их в исходный файл. После закрытия файлов, исходный файл содержит уже исправленную информацию, не включающую данные, которые мы хотели удалить:
Перед завершением функции delcd() программа вызывает функцию getslots(), обновляющую массив номеров ячеек.
Итак, теперь у нас есть два файла, которые содержат одинаковые данные. Единственный недостаток этой процедуры состоит в том, что диск компьютера должен иметь достаточно свободного места для хранения двух файлов с данными. Временный файл можно рассматривать как резервную копию. Если с исходным файлом произойдет что-то нежелательное, останется временный файл, содержащий данные в том виде, в каком они были до выполнения последней операции. Если вы по каким-либо причинам не хотите хранить временный файл на диске, то можете открыть его с режимом доступа "w", а потом немедленно закрыть:
tp = fopen(tempfile, "w");fclose(tp);В результате файл останется на диске, но совершенно пустой, так что он займет очень мало дискового пространства.
Программа продолжает проверку картотеки даже после того, как искомая запись удалена. Таким образом, вы можете удалить несколько одноименных записей.
|
Можно несколько усилить процедуру, если выводить на экран указанную запись и просить пользователя дать подтверждение, что эту запись действительно необходимо удалить. Если пользователь решит не удалять запись, программа должна скопировать ее во временный файл.