Работа с файлами
В этом разделе мы рассмотрим функции MS-DOS, предназначенные для создания, переименования, удаления и перемещения файлов. Операции чтения из файла и записи в файл будут описаны в следующем разделе.
Для создания файла предназначена функция 3Ch
прерывания INT 21h. С помощью этой функции может быть создан файл как в текущем, так и в любом другом каталоге. Если файл с указанным именем уже существует, он обрезается до нулевой длины. Будьте осторожны при использовании этой функции - она может уничтожить нужный вам файл.
Дополнительно функция 3Ch выполняет операцию открытия только что созданного файла, возвращая программе файловый индекс. При создании файла программа может указать требуемые атрибуты, которые будут использованы другими функциями для определения возможности предоставления доступа к файлу.
Приведем формат вызова функции создания файла:
На входе: | AH = 3Ch |
CX = Атрибуты создаваемого файла: 00h - обычный файл; 01h - только читаемый файл; 02h - скрытый файл; 04h - системный файл. |
|
DS:DX = Адрес строки, содержащей путь создаваемого файла | |
На выходе: | AX = Код ошибки, если был установлен в 1
флаг переноса CF; Файловый индекс, если флаг переноса сброшен в 0. |
При выполнении этой функции возможны следующие ошибки:
Операционная система игнорирует попытки создания с помощью этой функции каталогов или метки диска.
Для того, чтобы случайно не уничтожить содержимое файла с таким же именем, как и создаваемый, программа может использовать функцию 5Bh. Эта функция проверяет заданный путь на предмет наличия указанного файла. Если такой файл уже существует, функция возвращает программе признак ошибки.
Формат вызова функции:
На входе: | AH = 5Bh |
CX = Атрибуты создаваемого файла: 00h - обычный файл; 01h - только читаемый файл; 02h - скрытый файл; 04h - системный файл. |
|
DS:DX = Адрес строки, содержащей путь создаваемого файла | |
На выходе: | AX = Код ошибки, если был установлен в 1 флаг переноса CF; Файловый индекс, если флаг переноса сброшен в 0. |
На входе: | AH = 5Ah |
CX = Атрибуты создаваемого файла: 00h - обычный файл; 01h - только читаемый файл; 02h - скрытый файл; 04h - системный файл. |
|
DS:DX = Адрес буфера, в который функция запишет путь созданного временного файла. Размер этого буфера должен быть по крайней мере 13 байтов. | |
На выходе: | AX = Код ошибки, если был установлен в 1 флаг переноса CF; Файловый индекс, если флаг переноса сброшен в 0. |
Перед тем, как начать работу с файлом, его нужно открыть. Функции, создающие новые файлы, открывают новые файлы автоматически. Для того, чтобы открыть существующий файл, вы можете воспользоваться функцией 3Dh:
На входе: | AH = 3Dh |
AL = Требуемый режим доступа: Бит 7: флаг наследования 0 - файловый индекс наследуется порожденным процессом 1 - файловый индекс не наследуется порожденным процессом Биты 4...6: режим разделения 000 - режим совместимости 001 - запрещение всех видов доступа 010 - запрещение записи 011 - запрещение чтения 100 - разрешение всех видов доступа Бит 3:0 - зарезервировано Биты 0...2: вид доступа 000 - чтение 001 - запись 010 - чтение/запись |
|
DS:DX = Адрес строки, содержащей путь открываемого файла | |
На выходе: | AX = Код ошибки, если флаг переноса CF был установлен в 1; Файловый индекс, если флаг переноса сброшен в 0. |
Для использования битов 4...7 (управляющих доступом к файлу другими программами в сети) должна быть запущена программа SHARE.EXE.
Если используется бит наследования, то порожденному процессу наследуются запрошенный при открытии файла вид доступа.
Операционная система MS-DOS версии 4.0 имеет в своем составе функцию 6Ch, обладающую расширенными возможностями по созданию и открытию файлов:
На входе: | AH = 6Ch |
AL = 00h | |
BX = байт флагов расширенного режима открытия файла | |
CX = атрибуты создаваемого файла, используется только при создании файлов | |
DX = выполняемая функция, если файл существует или не существует: Биты 0-3 регистра DX задают действие, если файл существует: 0000h - если файл существует, вернуть признак ошибки 0001h - если файл существует, открыть его 0002h - если файл существует, заместить и открыть его Биты 4-7 регистра DX задают действие, если файл не существует: 0000h - если файл не существует, вернуть признак ошибки 0001h - если файл не существует, создать и открыть его |
|
DS:SI = Адрес строки, содержащей путь открываемого файла | |
На выходе: | AX = Код ошибки, если флаг переноса CF был установлен в 1; Файловый индекс, если флаг переноса сброшен в 0. |
CX = Код выполненных действий: 0 - файл был открыт 1 - файл был создан и открыт 2 - файл был замещен и открыт |
Биты | Назначение |
0...2 | Режим доступа при чтении/записи |
3 | Зарезервировано, должно быть равно 0 |
4...6 | Режим разделения |
7 | Флаг наследования |
8...12 | Зарезервировано, должно быть равно 0 |
13 | 0 - Режим обычного использования обработчика критических ошибок INT24h 1 - Блокировка обработчика критических ошибок INT 24h. Для того, чтобы узнать причину ошибки, программа должна использовать функцию 59h прерывания INT 21h MS-DOS. |
14 | Управление буферизацией: 0 - Использование стандартной для MS-DOS буферизации. 1 - Отмена буферизации. Использование этого режима замедлит работу с диском, однако вероятность потери информации при аварии в питающей сети уменьшится. |
Описанная выше функция является как бы комбинацией функций 3Dh и 3Ch (открытие и создание файла). Она удобна, но при ее использовании программа должна убедиться в том, что версия используемой операционной системы не ниже, чем 4.0.
Удалить файл можно при помощи функции 41h
прерывания INT 21h:
На входе: | AH = 41h |
DS:DX = Адрес строки в формате ASCIIZ, содержащей имя удаляемого файла. | |
На выходе: | AL = Код ошибки, если был установлен в 1 флаг переноса CF. |
Под удалением файла понимается вычеркивание файла из каталога и освобождение всех принадлежавших ранее этому файлу кластеров. Эти кластеры отмечаются в таблице размещения файлов как свободные. Уничтожения информации на диске при стирании файлов не происходит, поэтому в некоторых случаях можно полностью восстановить случайно удаленные файлы.
Стандартные библиотеки трансляторов Microsoft QC 2.5 и C 6.0 содержат функции для работы с файлами. Эти функции можно разделить на две группы - функции ввода/вывода низкого уровня и функции ввода/вывода потоком. Вторая группа функций использует буферизацию и будет рассмотрена в разделе, посвященном буферизованному вводу/выводу.
Функции ввода/вывода низкого уровня отображаются на описанные выше функции прерывания INT 21h (и функции этого же прерывания, предназначенные для чтения/записи, позиционирования и т.д.).
Для создания файла можно использовать функцию
creat():
int creat(char *filename, int mode);
Эта функция и ее параметры описаны в файлах io.h, sys\types.h, sys\stat.h, errno.h.
Первый параметр определяет путь создаваемого файла и его имя. Если файл с указанным именем существует, и не имеет атрибута "Только читаемый", функция сбрасывает длину файла до нуля. Предыдущее содержимое файла при этом уничтожается.
Второй параметр позволяет задать атрибуты создаваемого файла. Он может иметь следующие значения:
S_IWRITE | для создаваемого файла разрешена операция записи; |
S_IREAD | для создаваемого файла разрешена операция чтения; |
S_IREAD | S_IWRITE | для создаваемого файла разрешены операции чтения и записи. |
В операционных системах MS-DOS и OS/2 невозможно создать файл, в который можно было бы писать, но из которого было бы нельзя читать информацию. Поэтому задание второго параметра как S_IWRITE
приведет к созданию такого файла, для которого разрешены как операция записи, так и операция чтения.
После создания файла функция creat() открывает новый файл и возвращает файловый индекс (handle) или код ошибки.
Мощная функция open() предназначена как для открытия существующих файлов, так и для создания новых:
int open(char *filename, int oflag [, int pmode]);
Первый и третий параметры в этой функции аналогичны параметрам функции creat(), причем третий параметр нужен только при создании нового файла. Квадратные скобки указывают на то, что этот параметр является необязательным.
Параметр oflag может являться результатом битовой операции ИЛИ над следующими константами, определенными в файле fcntl.h:
O_APPEN | при записи в файл информация будет добавляться в конец файла; |
O_BINARY | файл открывается для работы в двоичном режиме (игнорируются управляющие символы, такие как конец строки); |
O_CREAT | создается новый файл и открывается для записи; эта константа игнорируется, если указанный в первом параметре файл уже существует; |
O_EXCL | используется вместе с O_CREAT; если указанный в первом параметре файл существует, функция возвратит признак ошибки; |
O_RDONLY | файл открывается только для чтения, попытка записи в файл приведет к тому, что функция записи вернет признак ошибки; |
O_RDWR | файл открывается как для чтения, так и для записи; |
O_TEXT | файл открывается в текстовом режиме; |
O_TRUNC | существующий файл открывается и обрезается до нулевой длины (если для этого файла разрешена операция записи); |
O_WRONLY | файл открывается только для записи (в операционных системах MS-DOS и OS/2 для файла, открытого с признаком O_WRONLY разрешено выполнение операции чтения). |
int close(int handle);
В качестве параметра функции передается файловый индекс, полученный при открытии или создании файла. Функция возвращает 0 при успешном закрытии файла, или -1 при ошибке.
Код ошибки для этой и других функций cтандартных библиотек трансляторов Microsoft QC 2.5 и C 6.0 записывается в глобальную переменную errno.