Основы программирования в Linux
Основы программирования в Linux читать книгу онлайн
Внимание! Книга может содержать контент только для совершеннолетних. Для несовершеннолетних чтение данного контента СТРОГО ЗАПРЕЩЕНО! Если в книге присутствует наличие пропаганды ЛГБТ и другого, запрещенного контента - просьба написать на почту [email protected] для удаления материала
Как и в случае
F_GETLK
l_start
l_whence
l_len
flock
fcntl
Команда
F_SETLKW
F_SETLK
Все блокировки файла, установленные программой, автоматически очищаются, когда закрывается соответствующий дескриптор файла. То же самое происходит, когда программа завершается.
Применение вызовов read и write при наличии блокировки
Когда вы применяете блокировку участков файла, очень важно использовать для доступа к данным низкоуровневые вызовы
read
write
fread
fwrite
fread
fwrite
fread
fread
read
Для того чтобы понять, в чем тут проблема, рассмотрим две программы, которые хотят обновить один и тот же файл. Предположим, что файл содержит 200 байтов данных, все нули. Первая программа начинает работу и устанавливает блокировку на запись для первых 100 байтов файла. Затем она применяет функцию
fread
fread
BUFSIZ
Затем стартует вторая программа. Она устанавливает блокировку
write
fread
read
write
Приведенное описание блокировки файла может показаться сложноватым, но ее труднее описать, чем применить. Поэтому выполните упражнение 7.9.
fcntl
Давайте рассмотрим пример работы блокировки файла в программе lock3.с. Для опробования блокировки вам понадобятся две программы: одна для установки блокировки и другая для ее тестирования. Первая программа выполняет блокировку.
1. Начните с файлов
include
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <fcntl.h>
const char *test_file = "/tmp/test_lock";
int main() {
int file desc;
int byte_count;
char *byte_to_write = "A";
struct flock region_1;
struct flock region_2;
int res;
2. Откройте файловый дескриптор:
file_desc = open(test_file, O_RDWR | O_CREAT, 0666);
if (!file_desc) {
fprintf(stderr, "Unable to open %s for read/writen", test_file);
exit(EXIT_FAILURE);
}
3. Поместите данные в файл:
for (byte_count = 0; byte_count < 100; byte_count++) {
(void)write(file_desc, byte_to_write, 1);
}
4. Задайте разделяемую блокировку для участка region 1 с 10-го байта по 30-й:
region_1.l_type = F_RDLCK;
region_1.l_whence = SEEK_SET;
region_1.l_start = 10;
region_1.l_len = 20;
5. Задайте исключительную блокировку для участка region_2 с 40-го байта по 50-й:
region_2.l_type = F_WRLCK;
region_2.l_whence = SEEK_SET;
region_2.l_start = 40;
region_2.l_len = 10;
6. Теперь заблокируйте файл:
printf("Process %d locking filen", getpid());
res = fcntl(file_desc, F_SETLK, &region_1);
if (res == -1) fprintf(stderr, "Failed to lock region 1n");
res = fcntl(file_desc, F_SETLK, &region_2);
if (res = fprintf(stderr, "Failed to lock region 2n");
7. Подождите какое-то время:
sleep(60);
printf ("Process %d closing filen", getpid());
close(file_desc);
exit(EXIT_SUCCESS);
}
Как это работает
Сначала программа создает файл, открывает его для чтения и записи и затем заполняет файл данными. Далее задаются два участка: первый с 10-го по 30-й байт для разделяемой блокировки и второй с 40-го по 50-й байт для исключительной блокировки. Затем программа выполняет вызов
fcntl