【转】linux文件读写 文件锁,Linux-文件IO_文件锁04

文件锁

这一节将讨论的是在文件已经共享

的情况下如何操作,也就是当多个程序共同操作一个文件的情况。Linux 中通常采用的方法是给文件上锁,

来解决对共享的资源的竞争。

文件锁包括建议性锁和强制性锁。建议性锁要求每个相关程序在访问文件之前检查是否有锁存在,并

且尊重已有的锁。一般情况下,不建议使用建议性锁,因为无法保证每个程序都自动检查是否有锁。而强

制性锁是由内核执行的锁,当一个文件被上锁进行写入操作的时候,内核将阻止其他任何程序对该文件进

行读写操作。采用强制性锁对性能的影响较大,每次读写操作内核都检查是否有锁存在。

在 Linux 中,实现文件上锁的函数有 lockf()和 fcntl(),其中 lockf()用于对文件施加建议性锁,而 fcntl()

不仅可以施加建议性锁,还可以施加强制性锁。同时,fcntl()还能对文件的某一记录上锁,也就是记录锁。

记录锁又可分为读取锁和写入锁,其中读取锁又称为共享锁,多个同时执行的程序允许在文件的同一

部分建立读取锁。而写入锁又称为排斥锁,在任何时刻只能有一个程序在文件的某个部分上建立写入锁。

显然,在文件的同一部分不能同时建立读取锁和写入锁。

fcntl()函数具有丰富的功能,它可以对已打开的文件进行各种操作。不仅能够管理文件锁,还可以获

取和设置文件相关标志位以及复制文件描述符等

所需头文件

#include

#include

#include

函数原型

int fcnt1(int fd, int cmd, …);

函数传入值

fd:文件描述符

cmd

F_GETLK:检测文件锁状态;

F_SETLK:设置 lock 描述的文件锁

F_SETLKW:这是 F_SETLK 的阻塞版本(命令名中的 W 表示等待(wait))

在无法获取锁时,会进入睡眠状态;如果可以获取锁或者捕捉到信号则会返回

函数返回值

成功:0

- 1:出错

如果 cmd 和锁操作相关,则第三个参数的类型为 struct *flock,其定义如下

struct flcok

{

short int l_type; /* 锁定的状态*/

short int l_whence;/*决定l_start位置*/

off_t l_start; /*锁定区域的开头位置*/

off_t l_len; /*锁定区域的大小*/

pid_t l_pid; /*锁定动作的进程*/

};

若要加锁整个文件,可以将 l_start 设置为 0,l_whence 设置为 SEEK_SET,l_len 设置为 0。

首先给 flock 结构体赋予相应的值,接着调用两次

fcntl()函数。第一次用 F_GETLK 命令判断是否可以执行 flock 结构所描述的锁操作:若成员 l_type 的值为

F_UNLCK,表示文件当前可以执行相应锁操作;否则成员 l_type 的值表示当前已有的锁类型并且成员 l_pid

被设置为拥有当前文件锁的进程号。

第二次用 F_SETLK 或 F_SETLKW 命令设置 flock 结构所描述的锁操作,后者是前者的阻塞版本。使

用后者时,当不能执行相应上锁/解锁操作时,程序会被阻塞,直到能够操作为止。

/* lock_set.c */

#include

#include

#include

#include

#include

int lock_set(int fd, int type)

{

struct flock old_lock, lock;

lock.l_whence = SEEK_SET;

lock.l_start = 0;

lock.l_len = 0;

lock.l_type = type;

lock.l_pid = -1;

#if 1

/* 判断文件是否可以上锁 */

fcntl(fd, F_GETLK, &lock);

if (lock.l_type != F_UNLCK)

{

/* 判断文件不能上锁的原因 */

if (lock.l_type == F_RDLCK) /* 该文件已有读取锁 */

{

printf("Read lock already set by %d\n", lock.l_pid);

}

else if (lock.l_type == F_WRLCK) /* 该文件已有写入锁 */

{

printf("Write lock already set by %d\n", lock.l_pid);

}

}

/* l_type 可能已被F_GETLK修改过 */

lock.l_type = type;

/* 根据不同的type值进行阻塞式上锁或解锁 */

if ((fcntl(fd, F_SETLKW, &lock)) < 0)

{

printf("Lock failed:type = %d\n", lock.l_type);

return 1;

}

switch(lock.l_type)

{

case F_RDLCK:

{

printf("Read lock set by %d\n", getpid());

}

break;

case F_WRLCK:

{

printf("Write lock set by %d\n", getpid());

}

break;

case F_UNLCK:

{

printf("Release lock by %d\n", getpid());

return 1;

}

break;

default:

break;

}/* end of switch */

#endif

return 0;

}

/* write_lock.c */

#include

#include

#include

#include

#include

int lock_set (int fd, int type);

int main(int argc, const char *argv[])

{

int fd;

if((fd = open("test.txt", O_RDWR | O_CREAT | O_TRUNC, 0666)) < 0)

{

perror("fail to open ");

return -1;

}

lock_set(fd, F_WRLCK);

getchar();

lock_set(fd, F_UNLCK);

getchar();

close(fd);

return 0;

}

同样开启两个终端,并首先启动终端一上的程序,其运行结果如下。

1:

2:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值