① 如果多个用户对一个文件进行操作的时候,如何解决,考虑用文件锁的形式和多路复用形式;
1)文件锁
找到一个函数flock()对文件进行加锁解锁等操作,就是在使用前对文件进行上锁,在使用后对文件进行解锁,这样就保证只有一个用户访问该文件。
flock()
表头文件 #include<sys/file.h>
定义函数 int flock(int fd,int operation);
函数说明 flock()会依参数operation所指定的方式对参数fd所指的文件做各种锁定或解除锁定的动作。此函数只能锁定整个文件,无法锁定文件的某一区域。
参数 operation有下列四种情况:
LOCK_SH 建立共享锁定。多个进程可同时对同一个文件作共享锁定。
LOCK_EX 建立互斥锁定。一个文件同时只有一个互斥锁定。
LOCK_UN 解除文件锁定状态。
LOCK_NB 无法建立锁定时,此操作可不被阻断,马上返回进程。通常与LOCK_SH或LOCK_EX 做OR(|)组合。
单一文件无法同时建立共享锁定和互斥锁定,而当使用dup()或fork()时文件描述词不会继承此种锁定。
返回值 返回0表示成功,若有错误则返回-1,错误代码存于errno。
代码:
#include <stdio.h>
#include <stdlib.h>
#include <sys/file.h>
int main(void)
{
FILE *fp = NULL;
int i = 20;
if ((fp = fopen("./file_lock.test", "r+b")) == NULL) //打开文件
printf("file open error!\n");
if (flock(fp->_fileno, LOCK_EX) != 0) //给该文件加锁
printf("file lock by others\n");
while(1) //进入循环,加锁时间为20秒,打印倒计时
{
printf("%d\n", i--);
sleep(1);
if (i == 0)
break;
}
fclose(fp); //20秒后退出,关闭文件
flock(fp->_fileno, LOCK_UN); //文件解锁
return 0;
}
#include <stdio.h>
#include <stdlib.h>
#include <sys/file.h>
int main(void)
{
FILE *fp = NULL;
int i = 0;
if ((fp = fopen("./file_lock.test", "r+b")) == NULL) //打开文件
printf("file open error!\n");
flock(fp->_fileno, LOCK_EX); //文件加锁
while(1) //进入循环
{
printf("%d\n", i++);
sleep(1);
}
fclose(fp); //关闭文件
flock(fp->_fileno, LOCK_UN); //释放文件锁
return 0;
}
首先运行file1.c,紧接着运行file2.c(运行file1.c后20秒内要运行file2.c否则看不到现象)
现象是:file1.c执行起来以后,开始倒计时。此时运行file2.c会阻塞在加锁处。当file1.c运行20秒后关闭文件,并释放文件锁后,file2.c会开始运行。
还找到一个fcndl()的函数,作用类似。
2)多路复用
1)文件锁
找到一个函数flock()对文件进行加锁解锁等操作,就是在使用前对文件进行上锁,在使用后对文件进行解锁,这样就保证只有一个用户访问该文件。
flock()
表头文件 #include<sys/file.h>
定义函数 int flock(int fd,int operation);
函数说明 flock()会依参数operation所指定的方式对参数fd所指的文件做各种锁定或解除锁定的动作。此函数只能锁定整个文件,无法锁定文件的某一区域。
参数 operation有下列四种情况:
LOCK_SH 建立共享锁定。多个进程可同时对同一个文件作共享锁定。
LOCK_EX 建立互斥锁定。一个文件同时只有一个互斥锁定。
LOCK_UN 解除文件锁定状态。
LOCK_NB 无法建立锁定时,此操作可不被阻断,马上返回进程。通常与LOCK_SH或LOCK_EX 做OR(|)组合。
单一文件无法同时建立共享锁定和互斥锁定,而当使用dup()或fork()时文件描述词不会继承此种锁定。
返回值 返回0表示成功,若有错误则返回-1,错误代码存于errno。
代码:
#include <stdio.h>
#include <stdlib.h>
#include <sys/file.h>
int main(void)
{
FILE *fp = NULL;
int i = 20;
if ((fp = fopen("./file_lock.test", "r+b")) == NULL) //打开文件
printf("file open error!\n");
if (flock(fp->_fileno, LOCK_EX) != 0) //给该文件加锁
printf("file lock by others\n");
while(1) //进入循环,加锁时间为20秒,打印倒计时
{
printf("%d\n", i--);
sleep(1);
if (i == 0)
break;
}
fclose(fp); //20秒后退出,关闭文件
flock(fp->_fileno, LOCK_UN); //文件解锁
return 0;
}
#include <stdio.h>
#include <stdlib.h>
#include <sys/file.h>
int main(void)
{
FILE *fp = NULL;
int i = 0;
if ((fp = fopen("./file_lock.test", "r+b")) == NULL) //打开文件
printf("file open error!\n");
flock(fp->_fileno, LOCK_EX); //文件加锁
while(1) //进入循环
{
printf("%d\n", i++);
sleep(1);
}
fclose(fp); //关闭文件
flock(fp->_fileno, LOCK_UN); //释放文件锁
return 0;
}
首先运行file1.c,紧接着运行file2.c(运行file1.c后20秒内要运行file2.c否则看不到现象)
现象是:file1.c执行起来以后,开始倒计时。此时运行file2.c会阻塞在加锁处。当file1.c运行20秒后关闭文件,并释放文件锁后,file2.c会开始运行。
还找到一个fcndl()的函数,作用类似。
2)多路复用
一直使用Select()或poll()函数监听需要读或者写的文件,若有文件准备好就进行读写操作。这样就可以实现多路复用。