找回密码
 用户注册

QQ登录

只需一步,快速开始

查看: 4988|回复: 0

读写锁算法

[复制链接]
发表于 2012-1-30 15:42:42 | 显示全部楼层 |阅读模式
背景:
在多线程编程中经常面临这样一个问题,同一个数据可以被多个线程同时读取,但是同一时间只能有一个线程对共享变量进行写操作。
对该问题常用的解决方法时声明一个锁(互斥量)来对共享变量进行保护,这样每次只能有一个线程来对数据进行读写,会导致读取的效率低下。
读写锁:
读写锁算法主要实现对共享资源访问时,可以在多个线程间同时进行读操作,但是在同一时间内只能有一个线程对共享资源进行修改,并且在写操作时不能有线程进行读操作。

算法思想:
当进行读操作时不能进行写操作,因此当有读操作时需要一把锁来锁住写操作,由于允许多个线程同时读操作,因此还需要一个变量(count)来记录当前读操作的线程个数,由于对count的修改需要互斥,因此还需要一个锁用来保存count的修改。

读写锁的数据结构:typedef struct RWLOCK_st
  1. {
  2. LOCK m_lReadLock;
  3. LOCK m_lWriteLock;
  4. int     m_icount;
  5. } RWLOCK;
复制代码
读写锁操作伪代码:
1. 声明一个全局的锁  RWLOCK  rwLock;
  1. rwLock.m_icount = 0;
复制代码


//读操作时加锁RWLock_LockRead()
  1. {
  2. rwLock.m_lReadLock->lock()
  3. rwLock.m_icount++;
  4. if (rwLock.m_icount  == 1)
  5. {
  6. rwLock.m_lWriteLock->lock(); //锁住写操作,当有读操作时只锁一次
  7. }
  8. rwLock.m_lReadLock->unlock()
  9. }
  10. RWLock_ReleaseRead()
  11. {
  12. rwLock.m_lReadLock->lock()
  13. rwLock.m_icount--;
  14. if (rwLock.m_icount  == 0)
  15. {
  16. rwLock.m_lWriteLock->unlock(); //允许写操作
  17. }
  18. rwLock.m_lReadLock->unlock()
  19. }
  20. RWLock_LockWrite()
  21. {
  22. rwLock.m_lWriteLock->lock() //锁住写操作
  23. }
  24. RWLock_ReleaseWrite()
  25. {
  26. rwLock.m_lWriteLock->unlock()
  27. }
  28. 从 RWLock_LockWrite()
  29. 和RWLock_ReleaseWrite()中可以看出当进行写操作时如果有读操作会阻塞在RWLock_ReleaseRead()函数rwLock.m_lWriteLock->lock()处,当一个线程完成写操作后,读操作会和写操作线程竞争rwLock.m_lWriteLock->lock()。
复制代码
读写锁缺点:
优点上面已经说过。
1. 进行读操作时会进行再次加锁和解锁操作,计算开销比较大,因此对锁内计算比较小的操作不适合使用读写锁。
2. 如果读操作比较密集,使得rwLock.m_icount永远不可能为0,因此会使写操作线程饿死。

参考:《多核计算与程序设计》,周伟明



作者:realxie 发表于2012-1-29 17:04:24 原文链接

您需要登录后才可以回帖 登录 | 用户注册

本版积分规则

Archiver|手机版|小黑屋|ACE Developer ( 京ICP备06055248号 )

GMT+8, 2024-12-22 17:19 , Processed in 0.014414 second(s), 6 queries , Redis On.

Powered by Discuz! X3.5

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表