概述

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库线程读写锁的更多相关文章

  1. freeswitch APR库

    概述 freeswitch依赖库源代码基本都可以在libs目录下找到. 在freeswitch的官方手册中,可以找到freeswitch的依赖库表格,其中freeswitch的core核心代码依赖库主 ...

  2. freeswitch APR库哈希表

    概述 freeswitch的核心源代码是基于apr库开发的,在不同的系统上有很好的移植性. 哈希表在开发中应用的非常广泛,主要场景是对查询效率要求较高的逻辑,是典型的空间换时间的数据结构实现. 大多数 ...

  3. freeswitch APR-UTIL库线程池实现分析

    概述 freeswitch的核心源代码是基于apr库开发的,在不同的系统上有很好的移植性. APR库在之前的文章中已经介绍过了,APR-UTIL库是和APR并列的工具库,它们都是由APACHE开源出来 ...

  4. Java线程读写锁

    排他锁和共享锁: 读写锁:既是排他锁,又是共享锁.读锁,共享锁,写锁:排他锁 读和读是不互斥的 import java.util.HashMap; import java.util.Map; impo ...

  5. Java 线程锁机制 -Synchronized Lock 互斥锁 读写锁

    (1)synchronized 是互斥锁: (2)ReentrantLock 顾名思义 :可重入锁 (3)ReadWriteLock :读写锁 读写锁特点: a)多个读者可以同时进行读b)写者必须互斥 ...

  6. Java基础-Java中的并法库之重入读写锁(ReentrantReadWriteLock)

    Java基础-Java中的并法库之重入读写锁(ReentrantReadWriteLock) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 在学习Java的之前,你可能已经听说过读 ...

  7. C#使用读写锁三行代码简单解决多线程并发写入文件时线程同步的问题

    (补充:初始化FileStream时使用包含文件共享属性(System.IO.FileShare)的构造函数比使用自定义线程锁更为安全和高效,更多内容可点击参阅) 在开发程序的过程中,难免少不了写入错 ...

  8. linux线程同步(3)-读写锁

    一.概述                                                    读写锁与互斥量的功能类似,对临界区的共享资源进行保护!互斥量一次只让一个线程进入临界区, ...

  9. 第8章 用户模式下的线程同步(3)_Slim读写锁(SRWLock)

    8.5 Slim读/写锁(SRWLock)——轻量级的读写锁 (1)SRWLock锁的目的 ①允许读者线程同一时刻访问共享资源(因为不存在破坏数据的风险) ②写者线程应独占资源的访问权,任何其他线程( ...

随机推荐

  1. k8s endpoints controller分析

    k8s endpoints controller分析 endpoints controller简介 endpoints controller是kube-controller-manager组件中众多控 ...

  2. selet 语句详解

    SELECT 语句的基本格式为:        SELECT 要查询的列名 FROM 表名字 WHERE 限制条件;        2.0 数学符号条件            SELECT 语句常常会 ...

  3. GIS应用|快速开发REST空间分析服务

    随着计算机的快速发展,GIS已经在各大领域得到应用,和我们的生活息息相关, 但是基于GIS几大厂商搭建服务,都会有一定的门槛,尤其是需要server,成本高,难度大,这里介绍一种在线GIS云平台,帮你 ...

  4. 1、使用ValueOperations操作redis(String字符串)

    文章来源:https://www.cnblogs.com/shiguotao-com/p/10559997.html 方法 c参数 s说明   void set(K key, V value); ke ...

  5. javascript-初级-day05js函数传参

    JS基础-01 函数传参.参数类型-1 <!DOCTYPE HTML> <html> <head> <meta http-equiv="Conten ...

  6. Codeforces 679E - Bear and Bad Powers of 42(线段树+势能分析)

    Codeforces 题目传送门 & 洛谷题目传送门 这个 \(42\) 的条件非常奇怪,不过注意到本题 \(a_i\) 范围的最大值为 \(10^{14}\),而在值域范围内 \(42\) ...

  7. HDU 6755 - Fibonacci Sum(二项式定理+推式子)

    题面传送门 其实是一道还好的题罢,虽然做了我 2147483647(bushi,其实是 1.5h),估计也只是因为 HDU 不支持数据下载所以错误总 debug 出来 首先看到 \(10^9+9\) ...

  8. R语言与医学统计图形-【24】ggplot位置调整函数

    ggplot2绘图系统--位置调整函数 可以参数position来调整,也有专门的函数position_*系列来设置. 位置函数汇总: 1.排列 并排排列 mean <- runif(12,1, ...

  9. python第三天 列表和元组

    枚举 for in enumerate 循环输出字符串的内容并且输出它的索引信息: #判断索引为5的字符串是不是"您" is in Python提供了⼤量的内置数据结构,包含了列表 ...

  10. 使用dumi生成react组件库文档并发布到github pages

    周末两天玩了下号称西湖区东半球最牛逼的react文档站点生成工具dumi,顺带结合github pages生成了react-uni-comps文档站, 一套弄下来,感觉真香,现在还只是浅尝,高级的特性 ...