读写锁是另一种实现线程间同步的方式。与互斥量类似,但读写锁将操作分为读、写两种方式,可以多个线程同时占用读模式的读写锁,这样使得读写锁具有更高的并行性。

读写锁的特性为:写独占,读共享;写锁优先级高。对于读写锁,掌握了这12个字就足矣了。

Linux环境下,读写锁具有以下三种状态:

  1. 读模式下加锁状态 (读锁)
  2. 写模式下加锁状态 (写锁)
  3. 不加锁状态

虽然读写锁有读锁、写锁、不加锁三种状态,但其实它只有一把锁,而非三把。

前文提到,读写锁的特性为:写独占,读共享;写锁优先级高。具体来讲:

  1. 读写锁是“写模式加锁”时, 解锁前,所有尝试对该锁进行加锁(不管是读锁还是写锁)的线程都会被阻塞;--> 写独占
  2. 读写锁是“读模式加锁”时, 如果线程以读模式对其加锁会成功;如果线程以写模式加锁会阻塞。--> 读共享
  3. 读写锁是“读模式加锁”时, 既有试图以写模式加锁的线程,也有试图以读模式加锁的线程。那么读写锁会阻塞随后的读模式锁请求,优先满足写模式锁。--> 写锁优先级高

读写锁也叫共享-独占锁。当读写锁以读模式锁住时,它是以共享模式锁住的;当它以写模式锁住时,它是以独占模式锁住的。写独占、读共享。

读写锁非常适合于对数据结构读的次数远大于写的情况。因为读锁是共享的,这样可以提高并行性。

主要应用函数:

pthread_rwlock_init函数

pthread_rwlock_destroy函数

pthread_rwlock_rdlock函数

pthread_rwlock_wrlock函数

pthread_rwlock_tryrdlock函数

pthread_rwlock_trywrlock函数

pthread_rwlock_unlock函数

以上7 个函数的返回值都是:成功返回0,失败直接返回错误号。

pthread_rwlock_t类型:用于定义一个读写锁变量,比如:pthread_rwlock_t rwlock;

pthread_rwlock_init函数

函数原型:

int pthread_rwlock_init(pthread_rwlock_t *restrict rwlock, const pthread_rwlockattr_t *restrict attr);

函数作用:

初始化一把读写锁

参数说明:

rwlock:传出参数,调用时应传&rwlock给该函数;

attr:表示读写锁属性,通常传NULL,表示使用默认属性;

pthread_rwlock_destroy函数

函数原型:

int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);

函数作用:

销毁一把读写锁

pthread_rwlock_rdlock函数

函数原型:

int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);

函数作用:

以读方式请求读写锁。(常简称为:请求读锁)

pthread_rwlock_wrlock函数

函数原型:

int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);

函数作用:

以写方式请求读写锁。(常简称为:请求写锁)

pthread_rwlock_unlock函数

函数原型:

int pthread_rwlock_unlock(pthread_rwlock_t *rwlock);

函数作用:

解锁。

pthread_rwlock_tryrdlock函数

函数原型:

int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock);

函数作用:

非阻塞以读方式请求读写锁(非阻塞请求读锁)

pthread_rwlock_trywrlock函数

函数原型:

int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock);

函数作用:

非阻塞以写方式请求读写锁(非阻塞请求写锁)

  1. /* 3个线程不定时 "写" 全局资源,5个线程不定时 "读" 同一全局资源 */
  2. #include <stdio.h>
  3. #include <unistd.h>
  4. #include <pthread.h>
  5. int counter; //全局资源
  6. pthread_rwlock_t rwlock;
  7. void *th_write(void *arg)
  8. {
  9. int t;
  10. int i = (int)arg;
  11. while (1) {
  12. t = counter;
  13. usleep(1000);
  14. pthread_rwlock_wrlock(&rwlock);
  15. printf("=======write %d: %lu: counter=%d ++counter=%d\n", i, pthread_self(), t, ++counter);
  16. pthread_rwlock_unlock(&rwlock);
  17. usleep(5000);
  18. }
  19. return NULL;
  20. }
  21. void *th_read(void *arg)
  22. {
  23. int i = (int)arg;
  24. while (1) {
  25. pthread_rwlock_rdlock(&rwlock);
  26. printf("----------------------------read %d: %lu: %d\n", i, pthread_self(), counter);
  27. pthread_rwlock_unlock(&rwlock);
  28. usleep(900);
  29. }
  30. return NULL;
  31. }
  32. int main(void)
  33. {
  34. int i;
  35. pthread_t tid[8];
  36. pthread_rwlock_init(&rwlock, NULL);
  37. for (i = 0; i < 3; i++)
  38. pthread_create(&tid[i], NULL, th_write, (void *)i);
  39. for (i = 0; i < 5; i++)
  40. pthread_create(&tid[i+3], NULL, th_read, (void *)i);
  41. for (i = 0; i < 8; i++)
  42. pthread_join(tid[i], NULL);
  43. pthread_rwlock_destroy(&rwlock); //释放读写琐
  44. return 0;
  45. }

更多精彩内容,请关注公众号良许Linux,公众内回复1024可免费获得5T技术资料,包括:Linux,C/C++,Python,树莓派,嵌入式,Java,人工智能,等等。公众号内回复进群,邀请您进高手如云技术交流群。


公众号:良许Linux

有收获?希望老铁们来个三连击,给更多的人看到这篇文章

Linux系统编程 —读写锁rwlock的更多相关文章

  1. 嵌入式 Linux线程同步读写锁rwlock示例

    读写锁比mutex有更高的适用性,可以多个线程同时占用读模式的读写锁,但是只能一个线程占用写模式的读写锁.1. 当读写锁是写加锁状态时,在这个锁被解锁之前,所有试图对这个锁加锁的线程都会被阻塞:2. ...

  2. Linux 系统编程 学习:11-线程:线程同步

    Linux 系统编程 学习:11-线程:线程同步 背景 上一讲 我们介绍了线程的属性 有关设置.这一讲我们来看线程之间是如何同步的. 额外安装有关的man手册: sudo apt-get instal ...

  3. linux系统编程之框架

    linux系统编程之框架: 1. 进程 1.1 进程概念 1.1.1 PCB 1.1.2 环境变量 1.2 进程控制 1.3 进程间通信 1.3.1 管道 1.3.2 有名管道 1.3.3 共享内存 ...

  4. linux系统编程之文件与io(五)

    上一节中已经学习了文件描述符的复制,复制方法有三种,其中最后一种fcntl还并未使用到,关于这个函数,不光只有复制文件描述符的功能,还有其它一些用法,本节就对其进行一一剖析: fcntl常用操作: 这 ...

  5. Linux系统编程温故知新系列 --- 01

    1.大端法与小端法 大端法:按照从最高有效字节到最低有效字节的顺序存储,称为大端法 小端法:按照从最低有效字节到最高有效字节的顺序存储,称为小端法 网际协议使用大端字节序来传送TCP分节中的多字节整数 ...

  6. Linux系统编程@进程通信(一)

    进程间通信概述 需要进程通信的原因: 数据传输 资源共享 通知事件 进程控制 Linux进程间通信(IPC)发展由来 Unix进程间通信 基于System V进程间通信(System V:UNIX系统 ...

  7. Linux 系统编程

    简介和主要概念 Linux 系统编程最突出的特点是要求系统程序员对它们工作的的系统的硬件和操作系统有深入和全面的了解,当然它们还有库和系统调用上的区别. 系统编程分为:驱动编程.用户空间编程和网络编程 ...

  8. 读书笔记之Linux系统编程与深入理解Linux内核

    前言 本人再看深入理解Linux内核的时候发现比较难懂,看了Linux系统编程一说后,觉得Linux系统编程还是简单易懂些,并且两本书都是讲Linux比较底层的东西,只不过侧重点不同,本文就以Linu ...

  9. Linux系统编程【转】

    转自:https://blog.csdn.net/majiakun1/article/details/8558308 一.Linux系统编程概论 1.1 系统编程基石 syscall: libc:标准 ...

随机推荐

  1. 微信小程序发送订阅消息(之前是模板消息)

    之前的模板消息已经废弃,现在改为订阅消息,订阅消息发布前,需要用户确认后才能接收订阅消息. 小程序端 index.wxml <button bindtap="send"> ...

  2. P2607 [ZJOI2008]骑士 基环树,树dp;

    P2607 [ZJOI2008]骑士 本题本质上就是树dp,和没有上司的舞会差不多,只不过多了一个对基环树的处理. #include<iostream> #include<cstri ...

  3. sql 游标(理论)

    游标是处理结果集的一种机制 --声明游标 --ISO 语法 DECLARE cursor_name [ INSENSITIVE ] [ SCROLL ] CURSOR FOR select_state ...

  4. 符合SEO的网站标题应该怎么写

    http://www.wocaoseo.com/thread-96-1-1.html 的seo网站标题既能提起读者的点击欲望,又能搜索引擎中获得好的排名,这两着之间有着有有一些联系,网站的标题若要从s ...

  5. package_ios

    PlistBuddy简单使用 https://www.jianshu.com/p/2167f755c47e xcodebuild 命令 https://www.jianshu.com/p/c32263 ...

  6. mac 下配置连接Linux服务器方法,上传下载文件操作

    1.先按照文档在本地生成SSHkey 2.mac输入 sudo -i 进入超级管理员#模式下,然后 创建用户 #useradd XXXadmin #passwd XXXadmin XXXadmin用户 ...

  7. 5 art-template

    npm 在终端命令下执行装包,就会在该目录下产生一个node_modules包 art-template地址:https://aui.github.io/art-template/zh-cn/docs ...

  8. axure rp extension for chrome怎么用

    1)打开文件(原型图)的resources文件夹>chrome文件夹>将文件axure-chrome-extension.crx拉至谷歌浏览器的扩展程序页面,点击确定添加 2)提示出错时, ...

  9. 20190925-05Redis五大数据类型之String 000 026

  10. Mybatis项目构建和CURD操作

    Mybatis入门 一.使用SqlSession对象创建Dao接口代理对象进行持久化操作 1.使用maven构建java项目 2.修改pom.xml配置,添加所需jar包坐标 <?xml ver ...