Linux应用开发 - 读写锁

什么时候用到读写锁:多线程开发中,如果少数线程会对共享内存数据修改,多数线程只是读取共享数据的值,就适用于读写锁解决“线程间抢夺资源”的问题。

  • 只要没有写模式下的加锁,任意线程都可以进行读模式下的加锁;
  • 只有读写锁处于不加锁状态时,才能进行写模式下的加锁;

当有多个线程发出读请求时,这些线程可以同时执行,也就说,共享数据的值可以同时被多个发出读请求的线程获取,当有多个线程发出写请求时,这些线程只能同步执行(互斥)。


本质读写锁就是一个全局变量

当读写锁发出读请求的线程占用时,称为读锁;当读写锁发出写请求的线程占用时,称为写锁

读写锁非常适合读数据的频率远大于写数据的频率从的应用中,这样可以在任何时刻运行多个读线程并发的执行,给程序带来了更高的并发度。

读写锁允许发出“读”请求的线程共享资源,发出“写”请求的线程必须独占资源,进而实现线程同步。

具体用法

读写锁用 pthread_rwlock_t 类型的变量表示,此类型定义在<pthread.h>头文件中

pthread_rwlock_t myRWLock;
  • 初始化读写锁

    • /*第一种*/
      pthread_rwlock_t myRWLock = PTHREAD_RWLOCK_INITIALIZER;
      
    • /*第二种*/
      int pthread_rwlock_init(pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *attr);
      
      • rwlock:初始化的读写锁变量
      • attr:用于自定义读写锁变量的属性,置为 NULL 时表示以默认属性初始化读写锁。
      • 返回值:成功:0;失败:非零数
  • 读锁

    • int pthread_rwlock_rdlock(pthread_rwlock_t* rwlock);
      int pthread_rwlock_tryrdlock(pthread_rwlock_t* rwlock);
      
      • pthread_rwlock_rdlock() 函数会阻塞当前线程,直至读写锁被释放;
      • pthread_rwlock_tryrdlock() 函数不会阻塞当前线程,直接返回 EBUSY。
  • 写锁

    • int pthread_rwlock_wrlock(pthread_rwlock_t* rwlock);
      int pthread_rwlock_trywrlock(pthread_rwlock_t* rwlock);
      
      • 当读写锁处于“无锁”状态时,两个函数都能成功获得写锁;当读写锁处于“读锁”或“写锁”状态时:

      • pthread_rwlock_wrlock() 函数会阻塞当前线程,直至读写锁被释放;

      • pthread_rwlock_trywrlock() 函数不会阻塞线程,直接返回 EBUSY。

  • 释放读写锁

    无论是处于“无锁”、“读锁”还是“写锁”的读写锁,都可以使用如下函数释放读写锁

    • int pthread_rwlock_unlock (pthread_rwlock_t* rwlock);
      
      • 由于多个线程可以同时获得“读锁”状态下的读写锁,这种情况下一个线程释放读写锁后,读写锁仍处于“读锁”状态,直至所有线程都释放读写锁,读写锁的状态才为“无锁”状态。

  • 销毁读写锁

    • int pthread_rwlock_destroy(pthread_rwlock_t* rwlock);
      
      • 返回值:成功:0;失败:非零数

案例

#include <pthread.h>
#include <stdio.h>
#include <unistd.h>
int x = 0;
//创建读写锁变量
pthread_rwlock_t myrwlock;void* read_thread(void* args){printf("------%u read_thread ready\n",pthread_self());while (1){sleep(1);//请求读锁pthread_rwlock_rdlock(&myrwlock);printf("read_thread: %u,x=%d\n", pthread_self(), x);sleep(1);//释放读写锁pthread_rwlock_unlock(&myrwlock);}return NULL;
}void* write_thread(void* param)
{printf("------%u write_thread ready!\n",pthread_self());while (1){sleep(1);// 请求写锁pthread_rwlock_wrlock(&myrwlock);++x;printf("write_thread: %u,x=%d\n", pthread_self(), x);sleep(1);//释放读写锁pthread_rwlock_unlock(&myrwlock);}return NULL;
}int main()
{int i;//初始化读写锁pthread_rwlock_init(&myrwlock, NULL);//创建 3 个读 x 变量的线程pthread_t readThread[3];for (i = 0; i < 3; ++i){pthread_create(&readThread[i], NULL, read_thread, NULL);}//创建 1 个修改 x 变量的线程pthread_t writeThread;pthread_create(&writeThread, NULL, write_thread, NULL);//等待各个线程执行完成pthread_join(writeThread, NULL);for (int i = 0; i < 3; ++i){pthread_join(readThread[i], NULL);}//销毁读写锁pthread_rwlock_destroy(&myrwlock);return 0;
}

运行结果:

------2450659072 write_thread ready!
------2459051776 read_thread ready
------2475837184 read_thread ready
------2467444480 read_thread ready
write_thread: 2450659072,x=1
read_thread: 2459051776,x=1
read_thread: 2467444480,x=1
read_thread: 2475837184,x=1
write_thread: 2450659072,x=2
read_thread: 2459051776,x=2
read_thread: 2467444480,x=2
read_thread: 2475837184,x=2

一共是四个线程(三个读锁线程,一个写锁线程)

读取x变量,读取变量前会先获得读锁写锁用于修改x变量的值,修改前先获得写锁

执行结果中三个读取x变量的线程总是能够获得x变量的值,因为能够同时过得读锁并同时执行

Linux应用开发 - 读写锁相关推荐

  1. linux内核之读写锁rwlock_t使用入门

    如果锁定写锁,则阻止其他地方读或者写,此时写或者读只能等待. 如果锁定写锁,则允许其他的读,但不允许写,写只能等待. 使用流程: 声明一个读写锁: rwlock_t mylock; 初始化: rwlo ...

  2. linux操作系统之读写锁

    (1)读写锁:只有一把锁,但是有两种状态(读,写) 1)读写锁的三种状态:读锁,写锁,不加锁 2)读写锁特性(12字):写锁优先级高,写独占,读共享 1>写模式加锁时,解锁前,所有对该线程加锁的 ...

  3. linux 进程间读写锁,Linux系统编程—进程间同步

    我们知道,线程间同步有多种方式,比如:信号量.互斥量.读写锁,等等.那进程间如何实现同步呢?本文介绍两种方式:互斥量和文件锁. ##互斥量mutex 我们已经知道了互斥量可以用于在线程间同步,但实际上 ...

  4. Linux线程同步读写锁 rwlock

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

  5. linux 进程 读写锁,linux 下实现高性能读写锁(read/write lock)

    前一篇文章分析了Windows slim read/write lock的工作原理.我们知道它的设计相当精妙,于是我们可以借鉴它的思路来设计linux下的读写锁. 在这个读写锁的设计上,需要注意的是l ...

  6. 深度解析Linux读写锁逻辑

    一.Linux为何会引入读写锁? 除了mutex,在linux内核中,还有一个经常用到的睡眠锁就是rw semaphore(后文简称为rwsem),它到底和mutex有什么不同呢?为何会有rw sem ...

  7. 【Linux内核】RW读写锁机制

    读写锁机制 Linux内核中读写锁的机制是一种多读单写的锁机制,它允许多个读操作同时进行,但只能有一个写操作进行.当有写操作时,所有读操作都会被阻塞,直到写操作完成. 在内核中,读写锁主要由以下两个结 ...

  8. 随想录(一种新的读写锁的写法)

    [ 声明:版权所有,欢迎转载,请勿用于商业用途.  联系信箱:feixiaoxing @163.com] 做游戏的同学想必对云风很熟悉.这一段时间他开发的skynet和ejoy2d两个软件框架在git ...

  9. go语言基础-----18-----协程安全、互斥锁、读写锁、匿名锁、sync.Once

    1 线(协)程安全-互斥锁 竞态检查工具是基于运行时代码检查,而不是通过代码静态分析来完成的,可以添加-race 来执行竞态检测.但是对于那些没 有机会运行到的代码逻辑中如果存在安全隐患,即使加了-r ...

最新文章

  1. 系统架构师学习笔记-面向对象方法
  2. 大型企业网络设备管理
  3. Python Poetry管理包安装速度慢的解决办法
  4. Solidity编程 五 之 数据类型
  5. Nicescroll滚动条插件手机端问题总结
  6. FLAG_ACTIVITY_CLEAR_TOP和singleTask的区别
  7. 统计学 贾俊平 笔记
  8. 倍市得聊体验:《梦华录》里的营销哲学
  9. java横向导出excel_利用Java进行Excel的数据导入导出
  10. 雷蛇公布2018年全年业绩:营收达7.1亿美元 净亏损9790万美元
  11. ffmpeg画中画效果
  12. 惊叹!前NASA员工绘大型精美地面立体画(高清组图)
  13. 计算机毕业设计Java乒乓球俱乐部管理(源码+系统+mysql数据库+lw文档)
  14. 图解 | 你管这破玩意儿叫TCP?
  15. tableau应用——某汽车销售分析
  16. 【Python】批量对文件做downsample并且以指定名称保存到文件夹中
  17. 为什么图吧导航SDK不免费
  18. html学习之input不可编辑
  19. 我所搜集的英语日语学习视频地址
  20. 一阶电路中的时间常数_怎样讲好一阶电路时间常数的概念

热门文章

  1. 麻衣神相 宋代古本 照片翻拍 全书87页
  2. 5分钟商学院-个人篇-演讲能力
  3. 经验总结木马免杀方法总结篇
  4. jmu–python–判断闰年_csf
  5. iOS 程序员职业发展:项目经理、技术经理还是产品经理
  6. NOI-1.4(07) 收集瓶盖赢大奖
  7. Windows11Ubuntu18.04安装OpenCV4.5.0VISP3.5.0
  8. vue + echarts 漏斗图 实现里面数据 外面标签 漏斗不随值改变而变形
  9. 自然语言处理NLP星空智能对话机器人系列:深入理解Transformer自然语言处理 GLUE Winograd schemas and NER
  10. BES(恒玄) 提示音解析