Операционная система MS-DOS© Александр Фролов, Григорий ФроловТом 1, книга 3, М.: Диалог-МИФИ, 1992. 3.5. Чтение/запись файловПосле того, как вы открыли файл, можно выполнять
над ним операции чтения/записи. Для записи данных
в файл предназначена функция 40h прерывания INT 21h.
В качестве параметров для этой функции
необходимо задать файловый индекс, полученный
при открытии существующего файла или создании
нового, адрес буфера, содержащего записываемые
данные и количество записываемых байтов:
При записи данные попадают в то место внутри файла, которое определяется содержимым так называемого файлового указателя позиции. При создании нового файла этот указатель сбрасывается в 0, что соответствует началу файла. При открытии файла с помощью функции 3Dh указатель также устанавливается на начало файла. Операция записи в файл с помощью функции 40h продвигает указатель вперед к концу файла на количество записываемых байтов. По мере увеличения размера файла ему будут распределяться все новые и новые кластеры из числа отмеченных как свободные. Если вам необходимо перезаписать содержимое файла, а не дописывать данные в конец, необходимо воспользоваться функцией позиционирования. Эта функция позволяет управлять содержимым файлового указателя позиции, она будет описана в следующем разделе. Следует учитывать, что количество действительно записанных байтов может не совпадать с заданным в регистре CX при вызове функции 40h. Такая ситуация возможна, например, при записи в файл, открытый в текстовом режиме, байта Ctrl-Z (1Ah). Этот байт означает конец текстового файла. Другая возможная причина - отсутствие свободного места на диске. Если функция вызывается с содержимым регистра CX, равным 0, файл будет обрезан или расширен до текущего положения файлового указателя. Разумеется, что если программа, выполняющая запись в файл, работает в сети, она должна иметь соответствующие права доступа к каталогу и файлу. Функция 40h может выполнять запись не только в файл, но и в устройство посимвольной обработки, предварительно открытое функцией 3Dh. Об этом мы говорили в разделах книги, посвященных драйверам. Для чтения данных из файла (или устройства
посимвольной обработки) предназначена функция 3Fh
прерывания INT 21h:
Эта функция используется аналогично функции записи. Для нее верны все замечания, касающиеся файлового указателя позиции, количества действительно прочитанных байтов и прав доступа. Если ваша программа составлена на языке программирования С, для записи и чтения данных она может воспользоваться функциями write() и read(): int write(int handle, void *buffer, unsigned count); int read(int handle, void *buffer, unsigned count); Эти функции работают аналогично функциям 40h и 3Fh прерывания INT 21h. Параметр handle определяет файл, для которого необходимо выполнить операцию записи или чтения. Параметр buffer - указатель на буфер, который содержит данные для записи или в который необходимо поместить прочитанные данные. Количество записываемых/читаемых байтов определяется третьим параметром - count. После выполнения операции функция возвращает количество действительно записанных или прочитанных данных или -1 при ошибке. Будьте внимательны, если вы записываете или читаете больше 32К байтов - вы можете получить признак ошибки, хотя передача данных выполнилась правильно. Большие массивы данных можно записывать по частям. В качестве примера мы приведем программу копирования файлов, которая пользуется описанными выше функциями ввода/вывода:
#include <io.h>
#include <conio.h>
#include <stdio.h>
#include <fcntl.h>
#include <sys\types.h>
#include <sys\stat.h>
#include <malloc.h>
#include <errno.h>
void main(int, char *[]);
void main(int argc, char *argv[]) {
int source, taget, i;
char *buffer;
unsigned count;
if(argc == 3) {
// Открываем исходный копируемый файл
if((source = open(argv[1], O_BINARY | O_RDONLY)) == - 1) {
printf("\nОшибка при открытии исходного файла: %d",
errno);
exit(-1);
}
// Открываем выходной файл. При необходимости создаем
// новый. Если файл уже существует, выводим на экран
// запрос на перезапись содержимого существующего файла
taget = open(argv[2], O_BINARY | O_WRONLY | O_CREAT | O_EXCL,
S_IREAD | S_IWRITE);
if(errno == EEXIST) {
printf("\nФайл существует. Перезаписать? (Y,N)\n");
// Ожидаем ответ оператора и анализируем его
i = getch();
if((i == 'y') || (i == 'Y'))
taget = open(argv[2], O_BINARY | O_WRONLY | O_CREAT | O_TRUNC,
S_IREAD | S_IWRITE);
}
// Если выходной файл открыть невозможно, выводим
// сообщение об ошибке и завершаем работу программы
if(taget == -1){
printf("\nОшибка при открытии выходного файла: %d",
errno);
exit(-1);
}
// Будем читать и писать за один раз 10000 байтов
count = 10000;
// Заказываем буфер для передачи данных
if((buffer = (char *)malloc(count)) == NULL) {
printf("\nНедостаточно оперативной памяти");
exit(-1);
}
// Копируем исходный файл
while(!eof(source)) {
// Читаем count байтов в буфер buffer
if((count = read(source, buffer, count)) == -1) {
printf("\nОшибка при чтении: %d",
errno);
exit(-1);
}
// Выполняем запись count байтов из буфера в выходной файл
if((count = write(taget, buffer, count)) == - 1) {
printf("\nОшибка при записи: %d",
errno);
exit(-1);
}
}
// Закрываем входной и выходной файлы
close(source);
close(taget);
// Освобождаем память, заказанную под буфер
free(buffer);
}
// Если при запуске программы не были указаны
// пути для входного или выходного файла,
// выводим сообщение об ошибке
else
printf("\n"
"Задайте пути для исходного"
" и результирующего файлов!\n");
}
В приведенной программе для определения конца исходного файла использована функция eof(): int eof(int handle); Для файла с файловым индексом handle эта
функция возвращает одно из трех значений:
Программа, которая читает файл с помощью функции 3Fh прерывания INT 21h, может определить момент достижения конца файла, анализируя код ошибки в регистре AX. |

