Майкл Джонсон - Разработка приложений в среде Linux. Второе издание
- Название:Разработка приложений в среде Linux. Второе издание
- Автор:
- Жанр:
- Издательство:Вильямс
- Год:2007
- Город:Москва
- ISBN:978-5-8459-1143-8
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Майкл Джонсон - Разработка приложений в среде Linux. Второе издание краткое содержание
Книга известных профессионалов в области разработки коммерческих приложений в Linux представляет собой отличный справочник для широкого круга программистов в Linux, а также тех разработчиков на языке С, которые перешли в среду Linux из других операционных систем. Подробно рассматриваются концепции, лежащие в основе процесса создания системных приложений, а также разнообразные доступные инструменты и библиотеки. Среди рассматриваемых в книге вопросов можно выделить анализ особенностей применения лицензий GNU, использование свободно распространяемых компиляторов и библиотек, системное программирование для Linux, а также написание и отладка собственных переносимых библиотек. Изобилие хорошо документированных примеров кода помогает лучше усвоить особенности программирования в Linux.
Книга рассчитана на разработчиков разной квалификации, а также может быть полезна для студентов и преподавателей соответствующих специальностей.
Разработка приложений в среде Linux. Второе издание - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
11.2.2. Закрытие файлов
Одной из операций, которые одинаковы для файлов всех типов, является закрытие файла. Ниже показано, как закрыть файл.
#include
int close(int fd);
Очевидно, что это базовая операция. Однако есть один важный момент, касающийся закрытия файлов, о котором следует помнить — она может завершиться сбоем. Некоторые системы (в первую очередь, следует вспомнить сетевые файловые системы вроде NFS) не пытаются поместить последнюю порцию записываемых данных в файл до тех пор, пока он не будет закрыт. Если такая операция вызовет сбой (например, по причине недоступности удаленного хоста), то close()
вернет ошибку. Если ваше приложение пишет данные, но не синхронизирует записи (см. обсуждение O_SYNC
в следующем разделе), то вы всегда должны проверять результат закрытия файла. Если close()
дает сбой, то это значит, что обновленный файл поврежден самым непредсказуемым образом! К счастью подобное случается достаточно редко.
11.2.3. Открытие файлов в файловой системе
Хотя Linux предусматривает множество типов файлов, обычные файлы используются наиболее часто. Программы, конфигурационные файлы, файлы данных — все они подпадают под это определение, и многие приложения не могут (явно) использовать файлы любых других типов. Есть два способа открытия файла, который имеет ассоциированное с ним имя.
#include
#include
int open(char *pathname, int flags, mode_t mode);
int creat(char *pathname, mode_t mode);
Функция open()
возвращает файловый дескриптор, указывающий на pathname
. Если возвращенное значение меньше нуля, значит, произошла ошибка (как всегда, errno
содержит код ошибки). Аргумент flags
описывает тип доступа, который нужен вызывающему процессу, а также управляет различными атрибутами открытия и манипулирования файлом. Режим доступа всегда должен быть указан, и он может быть одним из следующих: O_RDONLY
, O_RDWR
либо O_WRONLY
, что запрашивает доступ, соответственно, только по чтению, по чтению и записи либо только по записи. С этим режимом может быть объединены логическим "И" следующие значения для управления прочей семантикой файлов.
O_CREAT |
Если файл еще не существует, создать его как обычный файл. |
O_EXCL |
Этот флаг должен использоваться только с O_CREAT . Если он указан, то open() дает сбой в случае существования файла. Этот флаг позволяет реализовать простую блокировку, но не надежен при использовании в сетевых файловых системах типа NFS (подробно о блокировке файлов рассказывается в главе 13). |
O_NOCTTY |
Открываемый файл не становится управляющим терминалом процесса (см. главу 10). Этот флаг имеет значение только тогда, когда процесс, не имеющий управляющего терминала, открывает устройство tty. Если же он указан в любом другом случае, этот флаг игнорируется. |
O_TRUNC |
Если файл уже существует, его содержимое отбрасывается, и его размер устанавливается равным 0. |
O_APPEND |
Все операции записи выполняются в конец файла, хотя произвольный доступ по чтению также разрешен. |
O_NONBLOCK |
Файл открывается в неблокирующем режиме. Операции с нормальными файлами всегда блокируются, потому что они работают с локальными жесткими дисками, имеющими предсказуемое время отклика, но операции на некоторых типах файлов требуют непредсказуемого времени для завершения. Например, чтение из канала, в котором нет данных, блокирует процесс чтения до тех пор, пока данные в нем не появятся. Если же специфицирован флаг O_NONBLOCK , вызов read() вместо блокирования вернет ноль байт. Файлы, на операции с которыми может понадобиться непредсказуемый объем времени, называются медленными файлами. ( Примечание . O_NDELAY — оригинальное имя O_NONBLOCK , теперь устаревшее.) |
O_SYNC |
Обычно ядро перехватывает операции записи и сбрасывает их на физическое устройство тогда, когда это удобно. Хотя такая реализация значительно повышает производительность, появляется также возможность потери данных, чем в том случае, когда они немедленно пишутся на диск. Если при открытии файла указан флаг O_SYNC , то все изменения в файле сохраняются на диске перед тем, как ядро возвращает управления процессу, выполняющему запись. Это очень важно для некоторых приложений, таких как системы управления базами данных, в которых принудительная запись используется для предотвращения повреждения данных в случае сбоя системы. |
Параметр mode
указывает права доступа для файла, если он создается и если он модифицируется текущей установкой umask
процесса. Если не указано O_CREAT
, то mode
игнорируется.
Функция creat()
в точности эквивалентна следующему вызову:
open(pathname, O_CREAT | O_WRONLY | O_TRUNC, mode)
Мы не используем creat()
в этой книге, потому что находим функцию open()
более простой для чтения и понимания.
11.2.4. Чтение, запись и перемещение
Хотя есть несколько способов читать и писать файлы, мы обсудим здесь только простейшие из них [42] readv() , writev() и mmap() обсуждаются в главе 13; sendmsg() и recvmsg() упоминаются в главе 17.
. Чтение и запись почти идентичны, поэтому рассмотрим их одновременно.
#include
size_t read(int fd, void * buf, size_t length);
size_t read(int fd, const void * buf, size_t length);
Обе функции принимают файловый дескриптор fd
, указатель на буфер buf
и длину буфера length
, read()
читает из файлового дескриптора и помещает полученные данные в буфер; write()
пишет length
байт из буфера в файл. Обе функции возвращают количество переданных байт, или -1
в случае ошибки (это означает, что ничего не было прочитано или записано).
Теперь, когда мы описали эти системные вызовы, рассмотрим простой пример, создающий файл hw
в текущем каталоге и записывающий в него строку "Добро пожаловать!".
1: /* hwwrite.с */
2:
3: #include
4: #include
5: #include
6: #include
7: #include
8:
9: int main(void) {
10: int fd;
11:
12: /* открыть файл, создавая его, если он не существовал, и удаляя
13: его содержимое в противном случае */
14: if ((fd = open("hw", O_TRUNC | O_CREAT | O_WRONLY, 0644)) < 0) {
15: perror("open");
16: exit(1);
17: }
18:
19: /* магическое число 18 - это кол-во символов, которые
20: будут записаны */
21: if (write(fd, "Добро пожаловать!\n", 18) != 18) {
22: perror("write");
23: exit(1);
24: }
25:
26: close(fd);
27:
28: return 0;
29: }
Ниже показано, что получится, если запустить hwwrite
.
Интервал:
Закладка: