设为首页 - 加入收藏 ASP站长网(Aspzz.Cn)- 科技、建站、经验、云计算、5G、大数据,站长网!
热搜: 手机 数据 公司
当前位置: 首页 > 服务器 > 搭建环境 > Windows > 正文

Linux下的进程间通信:共享存储(2)

发布时间:2019-05-08 16:31 所属栏目:117 来源:Marty Kalin
导读:这个程序首先声明了一个类型为 struct flock 的变量,它代表一个锁,并对它的 5 个域做了初始化。第一个初始化 lock.l_type = F_WRLCK; /* exclusive lock */ 使得这个锁为排斥锁(read-write)而不是一个共享锁(r

这个程序首先声明了一个类型为 struct flock 的变量,它代表一个锁,并对它的 5 个域做了初始化。第一个初始化

  1. lock.l_type = F_WRLCK; /* exclusive lock */

使得这个锁为排斥锁(read-write)而不是一个共享锁(read-only)。假如生产者获得了这个锁,则其他的进程将不能够对文件做读或者写操作,直到生产者释放了这个锁,或者显式地调用 fcntl,又或者隐式地关闭这个文件。(当进程终止时,所有被它打开的文件都会被自动关闭,从而释放了锁)

  • 上面的程序接着初始化其他的域。主要的效果是整个文件都将被锁上。但是,有关锁的 API 允许特别指定的字节被上锁。例如,假如文件包含多个文本记录,则单个记录(或者甚至一个记录的一部分)可以被锁,而其余部分不被锁。

  • 第一次调用 fcntl

    1. if (fcntl(fd, F_SETLK, &lock) < 0)

    尝试排斥性地将文件锁住,并检查调用是否成功。一般来说, fcntl 函数返回 -1 (因此小于 0)意味着失败。第二个参数 F_SETLK 意味着 fcntl 的调用不是堵塞的;函数立即做返回,要么获得锁,要么显示失败了。假如替换地使用 F_SETLKW(末尾的 W 代指等待),那么对 fcntl 的调用将是阻塞的,直到有可能获得锁的时候。在调用 fcntl 函数时,它的第一个参数 fd 指的是文件描述符,第二个参数指定了将要采取的动作(在这个例子中,F_SETLK 指代设置锁),第三个参数为锁结构的地址(在本例中,指的是 &lock)。

  • 假如生产者获得了锁,这个程序将向文件写入两个文本记录。

  • 在向文件写入内容后,生产者改变锁结构中的 l_type 域为 unlock 值:

    1. lock.l_type = F_UNLCK;

    并调用 fcntl 来执行解锁操作。最后程序关闭了文件并退出。

  • 示例 2. 消费者程序

    1. #include <stdio.h>
    2. #include <stdlib.h>
    3. #include <fcntl.h>
    4. #include <unistd.h>
    5.  
    6. #define FileName "data.dat"
    7.  
    8. void report_and_exit(const char* msg) {
    9. [perror][4](msg);
    10. [exit][5](-1); /* EXIT_FAILURE */
    11. }
    12.  
    13. int main() {
    14. struct flock lock;
    15. lock.l_type = F_WRLCK; /* read/write (exclusive) lock */
    16. lock.l_whence = SEEK_SET; /* base for seek offsets */
    17. lock.l_start = 0; /* 1st byte in file */
    18. lock.l_len = 0; /* 0 here means 'until EOF' */
    19. lock.l_pid = getpid(); /* process id */
    20.  
    21. int fd; /* file descriptor to identify a file within a process */
    22. if ((fd = open(FileName, O_RDONLY)) < 0) /* -1 signals an error */
    23. report_and_exit("open to read failed...");
    24.  
    25. /* If the file is write-locked, we can't continue. */
    26. fcntl(fd, F_GETLK, &lock); /* sets lock.l_type to F_UNLCK if no write lock */
    27. if (lock.l_type != F_UNLCK)
    28. report_and_exit("file is still write locked...");
    29.  
    30. lock.l_type = F_RDLCK; /* prevents any writing during the reading */
    31. if (fcntl(fd, F_SETLK, &lock) < 0)
    32. report_and_exit("can't get a read-only lock...");
    33.  
    34. /* Read the bytes (they happen to be ASCII codes) one at a time. */
    35. int c; /* buffer for read bytes */
    36. while (read(fd, &c, 1) > 0) /* 0 signals EOF */
    37. write(STDOUT_FILENO, &c, 1); /* write one byte to the standard output */
    38.  
    39. /* Release the lock explicitly. */
    40. lock.l_type = F_UNLCK;
    41. if (fcntl(fd, F_SETLK, &lock) < 0)
    42. report_and_exit("explicit unlocking failed...");
    43.  
    44. close(fd);
    45. return 0;
    46. }

    (编辑:ASP站长网)

    网友评论
    推荐文章
      热点阅读