为了解决多个进程对同一个文件进行读写产生冲突的问题,在Linux系统中提供了flock这一系统调用,能够实现对文件的读写保护,即文件锁的功能。
flock提供的文件锁是建议性锁,只有当进程中调用了flock函数并根据返回值进行判断,然后自行处理是否要继续读写,flock的锁才能够起到作用。如果一个进程不调用flock对文件上的锁进行判断,而是强行进行读写操作,flock不会起到任何作用。即一个进程可以忽略其他进程对文件加的flock锁。
flock调用提供了两种类型的锁:
LOCK_SH:共享锁
LOCK_EX:独占锁
linux中使用C++调用flock的函数原型如下:
#include
int flock(int fd, int operation);
当flock设置成功后会返回0,设置失败返回-1并将错误码保存到errno中。
flock支持阻塞模式和非阻塞模式,默认是阻塞模式,当一个进程对文件加锁时如果无法成功加锁进程会一直进行阻塞等待;可以调整为非阻塞模式,进程无法成功加锁时flock直接返回-1.将阻塞模式调整为非阻塞模式需要将operation参数与LOCK_NB做或操作即可。
int ret = flock(fd, LOCK_SH | LOCK_NB);
int ret = flock(fd, LOCK_EX | LOCK_NB);
对于同一个文件而言,多个进程可以添加共享锁,但是只能存在一个互斥锁。
下面有两个进程A和B,
A对文件file加共享锁,B对文件加共享锁,A、B执行都正常。
A对文件file加共享锁,B对文件加互斥锁,或者A对文件加互斥锁,B对文件加共享锁,或者A对文件加互斥锁,B对文件加互斥锁,这三种情况,都会造成阻塞或者flock返回-1.
下面是一个简单的调用flock对文件进行加锁和解锁的例子:
#include <iostream>
#include <fcntl.h>
#include <unistd.h>
#include <sys/file.h>
#include <errno.h>
using namespace std;
#define FILEPATH "/home/flock/t.txt"
int main()
{
int fd;
fd = open(FILEPATH, O_RDWR | O_CREAT, 0644);
if(fd < 0){
cout<<"can not create the file"<<endl;
return -1;
}
int ret = flock(fd, LOCK_EX | LOCK_NB);
cout<<errno<<endl;
cout<<ret<<endl;
char buf[4]="bc\n";
int n = write(fd, buf, 4);
cout<<n<<endl;
flock(fd, LOCK_UN);
close(fd);
}