freeswitch APR库线程读写锁

概述
freeswitch的核心源代码是基于apr库开发的,在不同的系统上有很好的移植性。
线程读写锁在多线程服务中有重要的作用。对于读数据比写数据频繁的服务,用读写锁代替互斥锁可以提高效率。
由于APR库是跨平台的,而不同平台上的系统接口有区别,所以在APR库中就有一个适配层目录libs\apr\include\arch\,该目录下有不同系统的头文件定义,包括aix、beos、netware、os2、os390、unix、win32。
在编译源代码文件的过程中,根据当前系统自动选择不同的文件目录来适配。
下面我们对apr库的线程读写锁实现做一个介绍。
环境
centos:CentOS release 7.0 (Final)或以上版本
freeswitch:v1.8.7
GCC:4.8.5
读写锁数据结构
apr库的线程读写锁源代码文件在libs/apr目录下,这里只列出了unix和win32的版本。
libs\apr\include\apr_thread_rwlock.h
libs\apr\include\arch\unix\apr_arch_thread_rwlock.h
libs\apr\locks\unix\thread_rwlock.c
libs\apr\include\arch\win32\apr_arch_thread_rwlock.h
libs\apr\locks\win32\thread_rwlock.c
读写锁结构体定义。
我们可以看出,在unix系统下直接使用了pthread线程库的读写锁实现。
libs\apr\include\arch\unix\apr_arch_thread_rwlock.h
struct apr_thread_rwlock_t {
apr_pool_t *pool;
pthread_rwlock_t rwlock;
};
在win32系统则是使用了临界区CRITICAL_SECTION、mutex和event协同实现。
libs\apr\include\arch\win32\apr_arch_thread_rwlock.h
struct apr_thread_rwlock_t {
apr_pool_t *pool;
HANDLE write_mutex;
HANDLE read_event;
LONG readers;
CRITICAL_SECTION read_section;
};
常用函数
查看源代码头文件libs\apr\include\apr_thread_rwlock.h。
apr_thread_rwlock_create //创建并初始化1个线程读写锁
apr_thread_rwlock_rdlock //获取1个共享读锁
apr_thread_rwlock_tryrdlock //非阻塞获取1个共享读锁
apr_thread_rwlock_wrlock //获取1个独占写锁
apr_thread_rwlock_trywrlock //非阻塞获取1个独占写锁
apr_thread_rwlock_unlock //解锁1个读写锁
apr_thread_rwlock_destroy //销毁读写锁
从接口定义我们可以看出,加读锁和加写锁是分开的接口,而解锁是同一个接口。
接口的实现我们简单介绍一下。
在unix系统下,读写锁的接口中直接调用了pthread库中对应的读写锁接口。
比如apr_thread_rwlock_create中调用了pthread_rwlock_init,apr_thread_rwlock_rdlock中调用了pthread_rwlock_rdlock等等,这里就不再详细展开了,有兴趣的可以自己查看一下pthread_rwlock_t读写锁的定义和实现。
在win32系统下,读写锁的接口中使用了readers、read_event、write_mutex和read_section几个变量共同实现了读写锁的功能。
伪代码-读锁(读写锁, 超时时间)
进入临界区read_section
等待信号write_mutex,失败或超时则返回错误
计数器readers+1
重置事件read_event,失败则返回错误
释放write_mutex,失败则返回错误
离开临界区read_section
返回成功
伪代码-写锁(读写锁, 超时时间)
等待信号write_mutex,失败或超时则返回错误
等待信号read_event,失败或超时则释放write_mutex,并返回错误
返回成功
伪代码-解锁(读写锁)
释放write_mutex
计数器readers-1
设置事件read_event
返回
说实话。。。win32的代码我没看懂,有没有win开发的高手给讲解一下。
比如,为什么只有读锁中有临界区read_section的操作,为什么write_mutex只有释放操作,为什么写锁中没有等待readers为0。。。
总结
apr库的线程读写锁的实现介绍完了。
读写锁的适用场景是多线程下读多写少的业务流程,以实现独占写和共享读的逻辑。
在合适的场景中,读写锁比互斥锁有更高的并发能力和效率。
另外,还有强读者和强写者俩种不同的形式,读者可以自己深入研究一下。
空空如常
求真得真
freeswitch APR库线程读写锁的更多相关文章
- freeswitch APR库
概述 freeswitch依赖库源代码基本都可以在libs目录下找到. 在freeswitch的官方手册中,可以找到freeswitch的依赖库表格,其中freeswitch的core核心代码依赖库主 ...
- freeswitch APR库哈希表
概述 freeswitch的核心源代码是基于apr库开发的,在不同的系统上有很好的移植性. 哈希表在开发中应用的非常广泛,主要场景是对查询效率要求较高的逻辑,是典型的空间换时间的数据结构实现. 大多数 ...
- freeswitch APR-UTIL库线程池实现分析
概述 freeswitch的核心源代码是基于apr库开发的,在不同的系统上有很好的移植性. APR库在之前的文章中已经介绍过了,APR-UTIL库是和APR并列的工具库,它们都是由APACHE开源出来 ...
- Java线程读写锁
排他锁和共享锁: 读写锁:既是排他锁,又是共享锁.读锁,共享锁,写锁:排他锁 读和读是不互斥的 import java.util.HashMap; import java.util.Map; impo ...
- Java 线程锁机制 -Synchronized Lock 互斥锁 读写锁
(1)synchronized 是互斥锁: (2)ReentrantLock 顾名思义 :可重入锁 (3)ReadWriteLock :读写锁 读写锁特点: a)多个读者可以同时进行读b)写者必须互斥 ...
- Java基础-Java中的并法库之重入读写锁(ReentrantReadWriteLock)
Java基础-Java中的并法库之重入读写锁(ReentrantReadWriteLock) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 在学习Java的之前,你可能已经听说过读 ...
- C#使用读写锁三行代码简单解决多线程并发写入文件时线程同步的问题
(补充:初始化FileStream时使用包含文件共享属性(System.IO.FileShare)的构造函数比使用自定义线程锁更为安全和高效,更多内容可点击参阅) 在开发程序的过程中,难免少不了写入错 ...
- linux线程同步(3)-读写锁
一.概述 读写锁与互斥量的功能类似,对临界区的共享资源进行保护!互斥量一次只让一个线程进入临界区, ...
- 第8章 用户模式下的线程同步(3)_Slim读写锁(SRWLock)
8.5 Slim读/写锁(SRWLock)——轻量级的读写锁 (1)SRWLock锁的目的 ①允许读者线程同一时刻访问共享资源(因为不存在破坏数据的风险) ②写者线程应独占资源的访问权,任何其他线程( ...
随机推荐
- Linux系统僵尸进程详解
大安好,我是良许. 本文我们将来讨论一下什么是僵尸进程,僵尸进程是怎么产生的,如何杀死一个僵尸进程. Linux中的进程是什么? 讲到进程,我们要先了解一下另一个概念:程序. 程序说白了就是躺在电脑硬 ...
- CODING添加ssh提示格式错误的问题
不能去.shh文件夹打开id_rsa.pub文件查看 解决方法: 进入.ssh文件夹,然后右键git bash here 输入代码 cat id_rsa.pub 回车即可
- eclipse查看jar包源代码
方法一:将jd-gui集成在Eclipse中 https://jingyan.baidu.com/article/b24f6c8275536686bfe5daed.html 下载2个反编译文件, ...
- [luogu1737]旷野大计算
- 通过 for 循环,比较 Python 与 Ruby 编程思想的差别
作者:Doug Turnbull 译者:豌豆花下猫@Python猫 原文:https://softwaredoug.com/blog/2021/11/12/ruby-vs-python-for-loo ...
- 【模板】滑动窗口最值(单调队列)/洛谷P1886
题目链接 https://www.luogu.com.cn/problem/P1886 题目大意 有一个长为 \(n\) 的序列 \(a\) ,以及一个大小为 \(k\) 的窗口.现在这个从左边开始向 ...
- UE4类型数据自动注册
Version:4.26.2 UE4 C++工程名:MyProject 在<宏GENERATED_BODY做了什么?>中,简单分析了GENERATED_BODY宏给一个简单的.继承自UOb ...
- 《手把手教你》系列技巧篇(四十七)-java+ selenium自动化测试-判断元素是否显示(详解教程)
1.简介 webdriver有三种判断元素状态的方法,分别是isEnabled,isSelected 和 isDisplayed,其中isSelected在前面的内容中已经简单的介绍了,isSelec ...
- C语言中的除法的计算
不用除号,计算除法运算.思路是使用减法运算!思路1:循环采用减法每次减去n,直到做完减法之后结果小于0为止 但是这样次数较大 如求100/3,需要次数为34次. 思路2:循环采用减法每次减去k,K的 ...
- Shell 打印文件的最后5行
目录 Shell 打印文件的最后5行 题解-awk 题解-tail Shell 打印文件的最后5行 经常查看日志的时候,会从文件的末尾往前查看,于是请你写一个 bash脚本以输出一个文本文件 nowc ...