原文:http://blog.chinaunix.net/uid-24922718-id-3155067.html
早上花了半天的时间研究了下记录锁fcntl函数的使用方法,在这记录下以作备份
记录锁的概念上有两种锁,一种是共享锁(读锁),一种是互斥锁(写锁),记录锁就是一种对不同进程打开同一个文件进行读写操作的保护机制,就是当有人在写的时候,别人就不能打开文件读这个文件里的内容,当有人在读的时候,就不能让别人打开这个文件来看。
简单的说其实就是在打开一个文件对其进行操作以前,可以对其加上读锁(当然,如果是加读锁的话就一定要一读的方式打开文件),一个打开的文件可以加上1个或多个读锁,加上锁之后,其余的进程想要通过对其进行写操作的时候就会被阻塞或是出错,这取决于对与fcntl函数参数的选择。
同理,当一个进程对文件以写的方式打开后并加上了写锁(写锁一个文件上只能加一个,所以叫其互斥锁),则其余的进程就不能对其进行读锁的添加,也就不能进行读操作。
还是给出代码来的清晰啊:
#include <stdio.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
int lock_file(int fd,int cmd,int type,off_t offset,int whence,off_t len)
{
struct flock lock;
lock.l_type = type;
lock.l_start = offset;
lock.l_whence = whence;
lock.l_len = len;
return (fcntl(fd,cmd,&lock));
}
int main(int argc,char **argv)
{
int fd;
fd = open("./data",O_WRONLY | O_CREAT,00777); //可以看出现在是写的方式打开文件
pid_t pid;
char buf[100]="wahahah";
char max[100]="xxxxxxx";
if((pid = fork())<0) //开出两个进程去同时对这个文件进行操作
{
printf("fork error\n");
}
else if(pid == 0){
//child 子进程对已用读方式打开的文件进行加写锁,并对其进行写操作
lock_file(fd,F_SETLK,F_WRLCK,0,SEEK_SET,0);
if(write(fd,buf,sizeof(buf))==-1)
{
printf("child write error\n");
}
sleep(10);
printf("start unlock\n");
int n =lock_file(fd,F_SETLK,F_UNLCK,0,SEEK_SET,0); //10秒后对其进行解锁操作
close(fd);
printf("n is %d\n",n);
//unlock_file(fd,F_UNLCK,0,SEEK_SET,0);
}
else{
int par_fd;
par_fd = open("./data",O_RDONLY); //父进程以读的方式打开同一个文件
sleep(1); //保证子进程先于父进程加锁
for(;;)
{
if(lock_file(par_fd,F_SETLK,F_RDLCK,0,SEEK_SET,0) == -1)
{
printf("parent lock error\n"); //会发现,当对该文件进行读操作时,fcntl会出错,并返回-1,没有设置成等待模式
continue;
}
printf("unlock success\n"); //在子进程解锁后父进程读加锁成功
}
//parent
}
return 0;
}