C语言中的读写锁pthread_rwlock_t及使用方法介绍

144 篇文章 6 订阅

读写锁是用来解决多个读者和一个写者操作同一数据的一致性问题的,读操作可以共享,写操作是排他的,读可以有多个在读,写只有唯一个在写,写的时候不允许读。

 

具有强读者同步和强写者同步两种形式

强读者同步:当写者没有进行写操作,读者就可以访问;

强写者同步:当所有写者都写完之后,才能进行读操作,读者需要最新的信息,一些事实性较高的系统可能会用到该所,比如定票之类的。

 

读写锁的操作:

读写锁的初始化:

        定义读写锁:          pthread_rwlock_t  m_rw_lock;

        函数原型:              pthread_rwlock_init(pthread_rwlock_t * ,pthread_rwattr_t *);

        返回值:0,表示成功,非0为一错误码

读写锁的销毁:

        函数原型:             pthread_rwlock_destroy(pthread_rwlock_t* );

        返回值:0,表示成功,非0表示错误码

获取读写锁的读锁操作:分为阻塞式获取和非阻塞式获取,如果读写锁由一个写者持有,则读线程会阻塞直至写入者释放读写锁。

        阻塞式:

                            函数原型:pthread_rwlock_rdlock(pthread_rwlock_t*);

        非阻塞式:

                            函数原型:pthread_rwlock_tryrdlock(pthread_rwlock_t*);

       返回值: 0,表示成功,非0表示错误码,非阻塞会返回ebusy而不会让线程等待

获取读写锁的写锁操作:分为阻塞和非阻塞,如果对应的读写锁被其它写者持有,或者读写锁被读者持有,该线程都会阻塞等待。

      阻塞式:

                           函数原型:pthread_rwlock_wrlock(pthread_rwlock_t*);

      非阻塞式:

                           函数原型:pthread_rwlock_trywrlock(pthread_rwlock_t*);

       返回值: 0,表示成功

释放读写锁:

                         函数原型:pthread_rwlock_unlock(pthread_rwlock_t*);

总结(转):

互斥锁与读写锁的区别:

 

当访问临界区资源时(访问的含义包括所有的操作:读和写),需要上互斥锁;

当对数据(互斥锁中的临界区资源)进行读取时,需要上读取锁,当对数据进行写入时,需要上写入锁。

读写锁的优点:

对于读数据比修改数据频繁的应用,用读写锁代替互斥锁可以提高效率。因为使用互斥锁时,即使是读出数据(相当于操作临界区资源)都要上互斥锁,而采用读写锁,则可以在任一时刻允许多个读出者存在,提高了更高的并发度,同时在某个写入者修改数据期间保护该数据,以免任何其它读出者或写入者的干扰。

读写锁描述:

获取一个读写锁用于读称为共享锁,获取一个读写锁用于写称为独占锁,因此这种对于某个给定资源的共享访问也称为共享-独占上锁。

有关这种类型问题(多个读出者和一个写入者)的其它说法有读出者与写入者问题以及多读出者-单写入者锁。

  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在基于 rt-thread 的系统,可以使用 `struct rt_semaphore` 结构体来实现读写锁的控制。具体实现可以参考以下代码: ``` struct rt_semaphore rw_sem; int read_count; /* 初始化读写锁 */ int pthread_rwlock_init(pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *attr) { /* 初始化信号量,初始值为 1 */ rt_sem_init(&rw_sem, "rw_sem", 1, RT_IPC_FLAG_FIFO); read_count = 0; return 0; } /* 销毁读写锁 */ int pthread_rwlock_destroy(pthread_rwlock_t *rwlock) { /* 删除信号量 */ rt_sem_delete(&rw_sem); return 0; } /* 对读写锁进行读锁定 */ int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock) { /* 获取信号量,如果信号量值为 0,则线程阻塞 */ rt_sem_take(&rw_sem, RT_WAITING_FOREVER); read_count++; return 0; } /* 对读写锁进行写锁定 */ int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock) { /* 获取信号量,如果信号量值为 0,则线程阻塞 */ rt_sem_take(&rw_sem, RT_WAITING_FOREVER); while (read_count > 0) { /* 如果有读线程在使用,则等待读线程全部释放 */ rt_sem_release(&rw_sem); rt_sem_take(&rw_sem, RT_WAITING_FOREVER); } return 0; } /* 对读写锁进行解锁 */ int pthread_rwlock_unlock(pthread_rwlock_t *rwlock) { /* 释放信号量 */ rt_sem_release(&rw_sem); read_count--; return 0; } ``` 在上述代码,我们使用了一个信号量来实现读写锁的控制。信号量的初始值为 1,表示读写锁是未锁定状态。当线程需要对读写锁进行锁定时,首先需要获取信号量,如果信号量的值为 0,则线程阻塞等待。对于读锁,只需要将读计数器 `read_count` 加 1 即可;对于写锁,则需要判断是否有读线程在使用,如果有,则等待读线程全部释放后再获取锁。对于解锁操作,只需要释放信号量并将读计数器减 1 即可。 上述代码的函数名和参数与 POSIX 标准定义的函数名和参数保持一致,因此可以直接调用这些函数来实现读写锁的控制。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值