Операционная система UNIX - страница 57

стр.

>

>/* Выполним слияние потоков */

>dup2(fd, 2);

>...

Функция lseek(2)

С файловым дескриптором связан файловый указатель, определяющий текущее смещение в файле, начиная с которого будет произведена последующая операция чтения или записи. В свою очередь каждая операция чтения или записи увеличивают значение файлового указателя на число считанных или записанных байт. При открытии файла, файловый указатель устанавливается равным 0 или, если указан флаг >O_APPEND, равным размеру файла. С помощью функции lseek(2) можно установить файловый указатель на любое место файла и тем самым обеспечить прямой доступ к любой части файла. Функция имеет следующий вид:

>#include

>off_t lseek(int fildes, off_t offset, int whence);

Интерпретация аргумента >offset зависит от аргумента >whence, который может принимать следующие значения:

SEEK_CURУказатель смещается на >offset байт от текущего положения
SEEK_ENDУказатель смещается на >offset байт от конца файла
SEEK_SETУказатель устанавливается равным >offset

В случае успеха функция возвращает положительное целое, равное текущему значению файлового указателя.

Относительно системного вызова lseek(2) необходимо сделать два замечания. Во-первых, lseek(2) не инициирует никакой операции ввода/вывода, лишь изменяя значения файлового указателя в файловой таблице ядра. Во-вторых, смещение, указанное в качестве аргумента lseek(2), может выходить за пределы файла. В этом случае, последующие операции записи приведут к увеличению размера файла и, в то же время, к образованию дыры — пространства, формально незаполненного данными. В реальности, дыры заполняются нулями, но могут в ряде случаев привести к неприятным последствиям, с причиной и описанием которых вы сможете ознакомиться в главе 4 при обсуждении внутренней структуры файла.

Функция read(2) и readv(2)

Функции read(2) и readv(2) позволяют считывать данные из файла, на который указывает файловый дескриптор, полученный с помощью функций open(2), creat(2), dup(2), dup2(2), pipe(2) или fcntl(2). Функции имеют следующий вид:

>#include


>ssize_t read(int fildes, void *buf, size_t nbyte);


>#include

>#include


>ssize_t readv(int fildes, struct iovec *iov, int iovcnt);

Аргументы, передаваемые функции read(2), указывают, что следует считать >nbyte байт из файла, связанного с дескриптором >fildes, начиная с текущего значения файлового указателя. Считанные данные помещаются в буфер приложения, указатель на который передается в аргументе >buf. После завершения операции значение файлового указателя будет увеличено на >nbyte.

Функция readv(2) позволяет выполнить >iovcnt последовательных операций чтения за одно обращение к readv(2). Аргумент >iov указывает на массив структур, каждый элемент которого имеет вид:

>struct {

> void *iov_base; Указатель на начало буфера

> size_t iov_len; Размер буфера

>} iovec;

Функция readv(2) считывает данные из файла и последовательно размещает их в нескольких буферах, определенных массивом >iov. Такой характер работы, проиллюстрированный на рис. 2.8, получил название scatter read (от scatter (англ.) — разбрасывать). Общее число считанных байт в нормальной ситуации равно сумме размеров указанных буферов.

Рис. 2.8. Чтение файла с использованием нескольких буферов

Функции write(2) и writev(2)

Функции write(2) и writev(2) очень похожи на функции read(2) и readv(2), но используются для записи данных в файл. Функции имеют следующий вид:

>#include


>ssize_t write(int fildes, void *buf, size_t nbyte);


>#include

>#include


>ssize_t writev(int fildes, struct iovec *iov, int iovcnt);

Аргументы, передаваемые функции write(2), указывают, что следует записать >nbyte байт в файл, связанный с дескриптором >fildes, начиная с текущего значения файлового указателя. Данные для записи находятся в буфере приложения, указанном аргументом >buf. После завершения операции значение файлового указателя будет увеличено на >nbyte.

Аналогично функции readv(2), функция writev(2) позволяет выполнить >iovcnt последовательных операций записи за одно обращение к writev(2).