概述

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. 解决一个C#中定时任务被阻塞问题

    解决一个C#中定时任务被阻塞问题 目录 解决一个C#中定时任务被阻塞问题 1.摘要 2.C#中定时任务的最简方法 3.定时任务阻塞现象 4.阻塞现象原因分析 5.问题解决 1.摘要 本文会介绍一个C# ...

  2. 移动GPU分类/百科

    ARM mali gpu四大微架构概述 https://zhuanlan.zhihu.com/p/107141045 http://www.neardi.com/news_23/487.html

  3. C#简单配置类及数据绑定

    目录 简介 配置基类 派生配置类 数据绑定 Winform中的数据绑定 WPF下的数据绑定 附件 简介 本文实现一个简单的配置类,原理比较简单,适用于一些小型项目.主要实现以下功能: 保存配置到jso ...

  4. PTA7-2 愿天下有情人都是失散多年的兄妹

    呵呵.大家都知道五服以内不得通婚,即两个人最近的共同祖先如果在五代以内(即本人.父母.祖父母.曾祖父母.高祖父母)则不可通婚.本题就请你帮助一对有情人判断一下,他们究竟是否可以成婚? 输入格式: 输入 ...

  5. Python如何格式化输出

    目录 Python中的格式化输出 1.旧格式化 2.新格式format( ) 函数 Python中的格式化输出 格式化输出就是将字符串中的某些内容替换掉再输出就是格式化输出 旧格式化输出常用的有%d( ...

  6. 菜鸡的Java笔记 Eclipse 的使用

    Eclipse 的使用    1. Eclipse 简介    2. Eclipse 中的JDT 的使用    3. Eclipse 中的使用 junit 测试        Eclipse (中文翻 ...

  7. vs2012换肤功能,vs2012主题及自定义主题

    vs2012 默认的主题只有浅色和深色两种,深色太黑了,浅色又太亮了!怎么办? 研究不少时间终于找到了怎么编辑vs2012的皮肤 现在开始吧 打开vs2012-工具--扩展和更新 在联机中搜索 The ...

  8. [luogu5464]缩小社交圈

    不难证明合法当且仅当满足一下两个条件: 1.每一个位置最多被覆盖两次(无环) 2.将选择的区间按左端点从小到大排序,对于每一个左端点,其之前的区间的最大右端点不小于其(连通) (关于第一个的充分性证明 ...

  9. mybatis一对多查询resultMap只返回了一条记录

    问题描述:因为领导的一个需求,需要用到使用resultMap,很久没使用了,结果就除了点意外.就记录下这个问题 准备两个类:author(作者)和book(书),数据库创建对应的author-> ...

  10. Devs--开源规则引擎介绍

    Devs Devs是一款轻量级的规则引擎. 开源地址:https://github.com/CrankZ/devs 基础概念 此规则引擎的基础概念有字段.条件.规则等. 其中字段组成条件,条件组成规则 ...