Algorithm-Pattern
  • Introduction
  • C & C++
    • C语言
      • C/C++编译器
      • 宏的使用
      • 编译过程
      • 指针 & 数组
      • 柔性数组
      • 函数指针 & 回调函数
      • C标准库之stdio
      • C标准库之string
      • C标准库之errno
      • C标准库之stdarg
      • C标准库之regex
    • C++基础语法
      • 自增(++) & 自减(--)
      • c语言到c++
      • 可变模板参数
      • 强制类型转换
      • C/C++类型转换的本质
      • 指针 & 引用
      • const的用法
      • static的用法
      • 重要的关键字(一)
      • 重要的关键字(二)
      • 内存申请和释放
      • 内联函数
      • 函数 & 运算符重载
      • 面向对象之封装
      • 构造函数 & 析构函数
      • 面向对象之继承
      • 面向对象之多态
      • 泛型编程
      • 异常
      • 再谈函数指针
    • C++并发编程
      • C++的锁
      • 并发与多线程
    • C++高级特性
      • 函数对象
      • 移动语义 & 完美转发
      • lambda表达式
      • RTTI技术
      • RAII技术
      • 智能指针
      • 模板的特化
      • C++静态库和动态库
      • 内存溢出和内存泄漏
    • STL基础
      • String
      • array/vector/list
      • deque/priority_queue
      • set/map
      • unordered_set/unordered_map
      • algorithm_1
      • functional
      • allocator
    • C++标准库
      • IO
      • Tuple
      • regex
      • bitset
      • numeric
    • STL深入源码
      • vector内部实现
      • deque内部实现
      • sort函数实现
    • 第三方库
      • JsonCpp
      • ProtoBuf
  • 数据结构
    • 线性表
    • 字符串
    • 栈和队列
    • 二叉树
    • 平衡二叉树
    • 平衡多路搜索树
    • 树结构的延申
    • 图
    • 二进制
    • 散列表
  • 算法基础
    • 排序算法
    • 查找算法
    • 数学问题
    • 并查集
    • 递归算法
    • 附加——主定理
    • Catalan数
  • 算法设计思想
    • 滑动窗口思想
    • BFS/DFS
    • 二分法
    • 回溯法
    • 贪心算法
    • 分治法
    • 动态规划
    • 分支限界算法
    • 有限状态机(FSM)
  • LeetCode系列
    • 死磕二叉树
    • 股票买卖问题
    • 打家劫舍问题
    • 跳跃游戏问题
    • 括号匹配问题
    • 石子游戏问题
    • 子序列问题
    • 数组 & 矩阵
    • 排列 & 组合
  • 经典算法问题
    • 几何问题
    • 区间问题
    • 背包问题
    • 石子堆问题
    • 表达式求值
  • 面试题
    • 数据结构和算法基础
    • 程序设计题
      • 实现双线程交替打印
      • C++实现读写锁
      • 实现阻塞队列
      • 实现环形队列
      • 实现线程池
      • 实现智能指针
      • 实现string类
      • 实现高性能local cache
      • 实现内存池
      • 生产者-消费者模型
      • 设计定时器
    • 经典的算法题
    • C++面试题总结
    • 面试算法题总结
由 GitBook 提供支持
在本页

这有帮助吗?

  1. 面试题
  2. 程序设计题

C++实现读写锁

读写锁实际是一种特殊的自旋锁,它把对共享资源的访问者划分成读者和写者,读者只对共享资源进行读访问,写者则需要对共享资源进行写操作。这种锁相对于自旋锁而言,能提高并发性,因为在多处理器系统中,它允许同时有多个读者来访问共享资源,最大可能的读者数为实际的逻辑CPU数。写者是排他性的,一个读写锁同时只能有一个写者或多个读者(与CPU数相关),但不能同时既有读者又有写者。 在读写锁保持期间也是抢占失效的。

如果读写锁当前没有读者,也没有写者,那么写者可以立刻获得读写锁,否则它必须自旋在那里,直到没有任何写者或读者。 如果读写锁没有写者,那么读者可以立即获得该读写锁,否则读者必须自旋在那里,直到写者释放该读写锁。

互斥原则:

  • 读-读能共存

  • 读-写不能共存

  • 写-写不能共存

实现方式:

1、使用互斥锁加条件变量实现

2、使用原子操作实现

// WfirstRWLock.h
#ifndef ALGORITHM_PATTERN_CODE_WFIRSTRWLOCK_H
#define ALGORITHM_PATTERN_CODE_WFIRSTRWLOCK_H
#include <mutex>
#include <condition_variable>

class WfirstRWLock {
public:
    WfirstRWLock() = default;
    ~WfirstRWLock() = default;

    void lock_read();
    void lock_write();
    void release_read();
    void release_write();

private:
    volatile size_t m_count_read = 0;
    volatile size_t m_count_write = 0;
    volatile bool m_write_flag{ false };
    std::mutex m_mutex;
    std::condition_variable m_cond_write;
    std::condition_variable m_cond_read;
};

void testRWLock();

#endif //ALGORITHM_PATTERN_CODE_WFIRSTRWLOCK_H

// WfirstRWLock.cpp
#include "../../include/tools/WfirstRWLock.h"
#include <iostream>
#include <chrono>
#include <thread>
#include <future>
#include <ctime>
using namespace std;

void WfirstRWLock::lock_read(){
    unique_lock<mutex> lock(m_mutex);
    m_cond_read.wait(lock, [this]()->bool{return m_count_write == 0;});
    ++ m_count_read;
}

void WfirstRWLock::lock_write(){
    unique_lock<mutex> lock(m_mutex);
    m_cond_write.wait(lock, [this]()->bool{return m_count_read == 0 && !m_write_flag;});
    ++ m_count_write;
    m_write_flag = true;
}

void WfirstRWLock::release_read(){
    unique_lock<mutex> lock(m_mutex);
    -- m_count_read;
    if(m_count_read == 0 && m_count_write > 0){
        m_cond_write.notify_one();
    }
}

void WfirstRWLock::release_write(){
    unique_lock<mutex> lock(m_mutex);
    -- m_count_write;
    if(m_count_write == 0){
        m_cond_read.notify_all();
    }else{
        m_cond_write.notify_one();
    }
    m_write_flag = false;
}
// 测试代码
#if 1
static WfirstRWLock wfirstRwLock;
static int number = 1;

void getVal(promise<int> &pro){
    wfirstRwLock.lock_read();
    pro.set_value(number);
    this_thread::sleep_for(std::chrono::milliseconds(30));
    wfirstRwLock.release_read();
}

void putVal(int val){
    wfirstRwLock.lock_write();
    number = val;
    this_thread::sleep_for(std::chrono::milliseconds(100));
    wfirstRwLock.release_write();
}

void testRWLock(){
    srand((unsigned)time(NULL));
    for(int i = 0; i < 30; i++){
        int k = rand() % 10;
        if(k < 6){
            std::promise<int> resPro;
            std::future<int> resObj = resPro.get_future();
            thread r(getVal, std::ref(resPro));
            cout << resObj.get() << endl;
            r.join();
        }else{
            thread w(putVal, k + 1);
            w.join();
        }
    }
}
#endif
上一页实现双线程交替打印下一页实现阻塞队列

最后更新于4年前

这有帮助吗?