C++模拟实现读写锁

news2025/1/18 10:55:57

文章目录

  • 一、读者写者问题
  • 二、读写锁
    • 1.读写锁的概念
    • 2.读写锁的设计
      • (1)成员变量
      • (2)构造函数和析构函数
      • (3)readLock函数
      • (4)readUnlock函数
      • (5)writeLock函数
      • (6)writeUnlock函数
    • 3.RWLock类代码
  • 三、测试读写锁

一、读者写者问题

在编写多线程的时候,有一种情况是非常常见的,那就是有些公共数据,它们被修改的机会很少,但是被读取的机会却很多。通常来说,在读取的时候会伴随着查找等操作,如果按照正常地加锁方式给这种场景下的代码加锁,会极大地降低我们的效率。所以我们可以设计读写锁来解决读者写者问题。

二、读写锁

1.读写锁的概念

我们可以举一个生活中的小例子来类比理解一下读写锁:

比如在班上要出黑板报的时候,黑板就是临界资源,负责在黑板上写内容的同学就是写者,而我们这些来来往往观看黑板报的吃瓜群众就是写者。那么我们可以确定下面几种关系:

  1. 有同学在写黑板报的时候,读黑板报的同学必须等写黑板报的同学写完才可以开始读;而有同学在读黑板报的时候,写黑板报的同学必须等读黑板报的同学读完才可以开始写。
  2. 黑板报一次只能允许有一个同学在写,如果多个同学同时写黑板报可能会覆盖其他同学的内容。
  3. 读黑板报的同学可以同时一起读,并没有任何限制。

上面的关系可以类比成读者写者问题,所以读写锁要保证的关系是:读者和写者互斥、写者和写者互斥、读者和读者没有关系不受限制。

当前锁状态读锁请求写锁请求
无锁可以可以
读锁可以不可以
写锁不可以不可以

2.读写锁的设计

基于上述介绍的关系,我们可以利用互斥锁和信号量设计出一套读写锁机制:

核心思路就是使用引用计数,当有读者申请成功读锁以后,读者计数器加一。当写者申请成功写锁以后,写者计数器加一。

(1)成员变量

首先我们这个类需要一个互斥锁和一个条件变量,然后我们需要一个计数器rd_cnt记录当前读者的数量,另一个计数器wr_cnt记录当前写者的数量。

pthread_mutex_t mtx;
pthread_cond_t cond;
int rd_cnt; // 等待读的数量
int wr_cnt; // 等待写的数量

(2)构造函数和析构函数

构造函数用来初始化计数器、互斥锁和条件变量,析构函数用来释放资源。

RWLock() : rd_cnt(0), wr_cnt(0)
{
	pthread_mutex_init(&mtx, nullptr);
	pthread_cond_init(&cond, nullptr);
}
~RWLock()
{
	pthread_cond_destroy(&cond);
	pthread_mutex_destroy(&mtx);
}

(3)readLock函数

readLock函数是加读锁函数,首先进来需要先给它加锁,然后循环判断当前是否还有写者在进行写操作,如果是那么读者就要等待写者的条件变量,等到写者完成了写操作才能继续读。当循环条件不满足时,意味着写者完成了写操作,读者被唤醒,读者数量加一,最后释放锁,就可以执行读取操作了。

这里的加锁解锁目的是保证rd_cnt++这一语句的原子性,防止多个读者同时对rd_cnt进行加一操作而出现错误。所以这里是可以保证读者和读者之间不受限制的,因为外部调用readLock函数成功之后才可以开始进行读取操作,除非当前有写者在进行写操作,否则读者不会阻塞。

而读者和写者之间的互斥关系是通过循环判断来实现的,如果当前有写者正在写数据,readLock函数就会阻塞在pthread_cond_wait(&cond, &mtx);这条语句中,只有执行完这条语句readLock函数才能继续向下执行,执行完readLock函数之后读者才能进行读操作。

void readLock()
{
	pthread_mutex_lock(&mtx);
	// cout << "加读锁成功" << endl;

	// 如果当前还有写者在进行写操作
	// 那么读者应该等待写者的条件变量
	// 写者完成了写操作才能读
	while (wr_cnt > 0)
	pthread_cond_wait(&cond, &mtx);

	// 读者数量加一
	rd_cnt++;

	pthread_mutex_unlock(&mtx);
	// cout << "解读锁成功" << endl;
}

(4)readUnlock函数

在读者读取完数据之后就应该调用readUnlock函数释放读锁,readUnlock函数首先也是需要加锁保证rd_cnt--这一语句的原子性。然后还需要判断一下当前读者的数量是否为0,如果是的话,就唤醒正在等待的写者,可以开始写操作了。

void readUnlock()
{
	pthread_mutex_lock(&mtx);
	// cout << "unlock 加读锁成功" << endl;

	// 读者数量减一
	rd_cnt--;
	// 如果当前读者的数量为0,就唤醒正在等待的写者
	if (rd_cnt == 0)
	pthread_cond_signal(&cond);

	pthread_mutex_unlock(&mtx);
	// cout << "unlock 解读锁成功" << endl;
}

(5)writeLock函数

writeLock函数是加写锁的函数,同样地,它也需要先加锁保证语句wr_cnt++语句的原子性。然后需要循环判断当前是否有读者或者有其它的写者在进行读写操作,如果有的话,只要有一个读者或者有一个写者,当前这个写者都不能进行写操作,必须等待其它读者或者写者完成之后才能进行。

void writeLock()
{
	pthread_mutex_lock(&mtx);
	// cout << "加写锁成功" << endl;

	// 如果当前有读者或者有其它写者在进行读写操作时
	// 当前的写者都不能进行写操作,必须等待
	while (wr_cnt >= 1 || rd_cnt >= 1)
	pthread_cond_wait(&cond, &mtx);

	// 走到这里,一定是当前既没有读者也没有写者在进行读写操作了
	// 此时当前的写者才能进行写操作
	// 写者数量加一
	wr_cnt++;

	pthread_mutex_unlock(&mtx);
	// cout << "解写锁成功" << endl;
    }

(6)writeUnlock函数

writeUnlock函数是解写锁函数,与前面的几个函数接口类似,只不过它需要判断一下当前的写者数量是否为0,如果是的话,就要唤醒另一个正在等待条件变量的写者。

void writeUnlock()
{
	pthread_mutex_lock(&mtx);
	// cout << "unlock 加写锁成功" << endl;

	// 写者数量减一
	wr_cnt--;
	// 如果当前写者数量为0,则唤醒正在等待的另一个写者
	if (wr_cnt == 0)
	pthread_cond_signal(&cond);

	pthread_mutex_unlock(&mtx);
	// cout << "unlock 解写锁成功" << endl;
}

3.RWLock类代码

#pragma once
#include <pthread.h>

using namespace std;

// 写者和写者之间是互斥关系
// 写者和读者之间是互斥关系
// 读者和读者之间没有关系

class RWLock
{
private:
    pthread_mutex_t mtx;
    pthread_cond_t cond;
    int rd_cnt; // 等待读的数量
    int wr_cnt; // 等待写的数量

public:
    RWLock() : rd_cnt(0), wr_cnt(0)
    {
        pthread_mutex_init(&mtx, nullptr);
        pthread_cond_init(&cond, nullptr);
    }
    ~RWLock()
    {
        pthread_cond_destroy(&cond);
        pthread_mutex_destroy(&mtx);
    }

    void readLock()
    {
        pthread_mutex_lock(&mtx);
        // cout << "加读锁成功" << endl;

        // 如果当前还有写者在进行写操作
        // 那么读者应该等待写者的条件变量
        // 写者完成了写操作才能读
        while (wr_cnt > 0)
            pthread_cond_wait(&cond, &mtx);

        // 读者数量加一
        rd_cnt++;

        pthread_mutex_unlock(&mtx);
        // cout << "解读锁成功" << endl;
    }

    void readUnlock()
    {
        pthread_mutex_lock(&mtx);
        // cout << "unlock 加读锁成功" << endl;

        // 读者数量减一
        rd_cnt--;
        // 如果当前读者的数量为0,就唤醒正在等待的写者
        if (rd_cnt == 0)
            pthread_cond_signal(&cond);

        pthread_mutex_unlock(&mtx);
        // cout << "unlock 解读锁成功" << endl;
    }

    void writeLock()
    {
        pthread_mutex_lock(&mtx);
        // cout << "加写锁成功" << endl;

        // 如果当前有读者或者有其它写者在进行读写操作时
        // 当前的写者都不能进行写操作,必须等待
        while (wr_cnt >= 1 || rd_cnt >= 1)
            pthread_cond_wait(&cond, &mtx);

        // 走到这里,一定是当前既没有读者也没有写者在进行读写操作了
        // 此时当前的写者才能进行写操作
        // 写者数量加一
        wr_cnt++;

        pthread_mutex_unlock(&mtx);
        // cout << "解写锁成功" << endl;
    }

    void writeUnlock()
    {
        pthread_mutex_lock(&mtx);
        // cout << "unlock 加写锁成功" << endl;

        // 写者数量减一
        wr_cnt--;
        // 如果当前写者数量为0,则唤醒正在等待的另一个写者
        if (wr_cnt == 0)
            pthread_cond_signal(&cond);

        pthread_mutex_unlock(&mtx);
        // cout << "unlock 解写锁成功" << endl;
    }
};

三、测试读写锁

我们用代码随机生成一份测试用例,测试数据文件包括 n 行测试数据,分别描述创建的 n 个线程是读者还是写者,以及读写操作的开始时间和持续时间。

每行测试数据包括四个字段,各个字段间用空格分隔。

第一字段为一个正整数,表示线程序号。

第二字段表示相应线程角色,R表示读者,W表示写者。

第三字段为一个正数,表示读写操作的开始时间,线程创建后,延迟相应时间(单位为秒)后发出对共享资源的读写申请。

第四字段为一个正数,表示读写操作的持续时间。当线程读写申请成功后,开始对共享资源的读写操作,该操作持续相应时间后结束,并释放共享资源。

测试数据随机生成,为简化起见,假设每个线程只执行一次读或写操作,之后运行结束。

下面是一个测试数据文件的例子:

在这里插入图片描述

测试代码:

#include <iostream>
#include <vector>
#include <string>
#include <set>
#include <cstdlib>
#include <cmath>
#include <ctime>
#include <pthread.h>
#include <unistd.h>
#include "ReaderAndWriter.hpp"

using namespace std;

RWLock rwlock; // 定义全局的读写锁

struct ThreadData
{
    int _id;
    vector<int> _continue_time;

    ThreadData(int id, vector<int> &continue_time)
        : _id(id), _continue_time(continue_time)
    {
    }
};

// 临界资源
// 读者负责读取,写者负责加一
int value = 100;

volatile static int count = 0; // 用来记录每个线程是否执行完毕

pthread_mutex_t count_mutex = PTHREAD_MUTEX_INITIALIZER;

void *reader(void *args)
{
    // 设置线程分离
    pthread_detach(pthread_self());
    ThreadData *thread_data = (ThreadData *)args;

    // 加读锁
    cout << "线程" << thread_data->_id + 1 << "发出读操作申请" << endl;
    rwlock.readLock();
    cout << "线程" << thread_data->_id + 1 << "开始读操作, value: " << value << endl;
    sleep(thread_data->_continue_time[thread_data->_id]);
    rwlock.readUnlock();
    cout << "线程" << thread_data->_id + 1 << "结束读操作" << endl;

    // 走到这里说明当前的读操作已经结束了,count加一,当count==n的时候说明所有线程已经执行完毕
    pthread_mutex_lock(&count_mutex);
    count++;
    pthread_mutex_unlock(&count_mutex);
}

void *writer(void *args)
{
    // 设置线程分离
    pthread_detach(pthread_self());
    ThreadData *thread_data = (ThreadData *)args;

    // 加写锁
    cout << "线程" << thread_data->_id + 1 << "发出写操作申请" << endl;
    rwlock.writeLock();
    cout << "线程" << thread_data->_id + 1 << "开始写操作" << endl;
    value++;
    sleep(thread_data->_continue_time[thread_data->_id]);
    rwlock.writeUnlock();
    cout << "线程" << thread_data->_id + 1 << "结束写操作" << endl;

    // 走到这里说明当前的读操作已经结束了,count加一,当count==n的时候说明所有线程已经执行完毕
    pthread_mutex_lock(&count_mutex);
    count++;
    pthread_mutex_unlock(&count_mutex);
}

void createReader(int id, vector<int> continue_time)
{
    pthread_t tid;
    ThreadData *thread_data = new ThreadData(id, continue_time);
    pthread_create(&tid, nullptr, reader, (void *)thread_data);
    cout << "线程" << id + 1 << "创建成功" << endl;
}

void createWriter(int id, vector<int> continue_time)
{
    pthread_t tid;
    ThreadData *thread_data = new ThreadData(id, continue_time);
    pthread_create(&tid, nullptr, writer, (void *)thread_data);
    cout << "线程" << id + 1 << "创建成功" << endl;
}

int main()
{
    srand(time(0));
    // n个测试用例
    int n;
    cin >> n;

    // 随机生成测试用例
    // 1.随机生成角色
    char stlect_str[2] = {'R', 'W'};
    vector<char> role;
    role.resize(n);
    for (int i = 0; i < n; i++)
    {
        int num = rand() % 2;
        role[i] = stlect_str[num];
    }

    // 2.随机生成开始读写时间
    set<int> begin_time;
    while (begin_time.size() < n)
    {
        // 限制随机数在20s以内
        int num = rand() % 20;
        begin_time.insert(num);
    }

    // 3.随机生成读写持续时间
    vector<int> continue_time;
    continue_time.resize(n);
    for (int i = 0; i < n; i++)
    {
        // 持续时间限制在5s以内
        int num = rand() % 5 + 1;
        continue_time[i] = num;
    }

    set<int>::iterator iter = begin_time.begin();
    for (int i = 0; i < n; i++)
    {
        cout << i + 1 << "\t" << role[i] << "\t" << *iter << "\t" << continue_time[i] << endl;
        iter++;
    }

    // 记录当前系统的时间
    time_t begin = time(nullptr);
    int id = 0;
    const double eps = 1e-8;
    iter = begin_time.begin();
    while (id < n)
    {
        time_t end = time(nullptr);
        // 到达开始时间,创建线程
        if (abs((*iter) - difftime(end, begin)) < eps)
        {
            // 创建读者线程
            if (role[id] == 'R')
            {
                createReader(id, continue_time);
            }
            // 创建写者线程
            else
            {
                createWriter(id, continue_time);
            }
            id++;
            iter++;
        }
    }
    //
    while (true)
    {
        if(count == n)
        {
            cout << "所有线程均已完成读写操作,主线程退出" << endl;
            break;
        }
    }
    return 0;
}

运行结果一:

在这里插入图片描述

运行结果二:

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/415115.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

为什么黑客不黑/攻击赌博网站?如何入门黑客?

攻击了&#xff0c;只是你不知道而已&#xff01; 同样&#xff0c;对方也不会通知你&#xff0c;告诉你他黑了赌博网站。 攻击赌博网站的不一定是正义的黑客&#xff0c;也可能是因赌博输钱而误入歧途的法外狂徒。之前看过一个警方破获的真实案件&#xff1a;28岁小伙因赌博…

Linux 操作系统原理作业 - 行人与机动车问题

大三上学期操作系统原理这门课中&#xff0c;老师给了一道作业《行人与机动车问题》&#xff1b; 即Linux多线程下处理行人与机动车谁优先的问题&#xff0c;需要用到多线程和互斥量&#xff1b; 行人 - 机动 车问题 假设有一个路口&#xff0c;有很多行人和机动车需要通过&a…

1673_MIT 6.828 Homework xv6 lazy page allocation要求翻译

全部学习汇总&#xff1a; GreyZhang/g_unix: some basic learning about unix operating system. (github.com) 在计划表中看到了这样一份作业&#xff0c;做一个简单的翻译整理。原来的页面&#xff1a;Homework: xv6 lazy page allocation (mit.edu) 家庭作业&#xff1a;x…

代码版本M、RC、GA、Release等标识的区别

引言 最近听说spring framework有了重大版本调整&#xff0c;出了6.0的GA版本了 那GA是啥意思呢&#xff1f; 看了下spring 官网和代码仓库&#xff0c;除了GA&#xff0c;还有M、RC、Release等 Spring FrameworkLevel up your Java code and explore what Spring can do f…

[Java Web]element | 一个由饿了么公司开发的前端框架,让你快速构建现代化、美观的 Web 应用程序。

⭐作者介绍&#xff1a;大二本科网络工程专业在读&#xff0c;持续学习Java&#xff0c;努力输出优质文章 ⭐作者主页&#xff1a;逐梦苍穹 ⭐所属专栏&#xff1a;Java Web ⭐如果觉得文章写的不错&#xff0c;欢迎点个关注一键三连&#x1f609;有写的不好的地方也欢迎指正&a…

【mybatis】mybatis的工作原理

目录一、工作流程二、说明2.1 构建SqlSessionFactory2.2 SqlSession的获取2.3 SqlSession执行语句三、源码结构3.1 接口层3.2 核心处理3.3 核心处理层四、代码示例4.1 通过inputStream构建SqlSessionFactory4.2 通过configuration构建SqlSessionFactory4.3 mybatis-config.xml示…

groovy环境搭建

什么是DSL? 领域特定语言DSL&#xff08;全称&#xff1a;domain specific language&#xff09; 常见的DSL语言有&#xff1a;UML、HTML、SQL、XML、Groovy 作用&#xff1a;解决某一特定领域的问题 什么是groovy? groovy是一种基于JVM的敏捷开发语言。 结合了Python、Ruby和…

Vite4+Vuejs3项目初步搭建,并部署多个vue项目到nginx

前提条件 1、熟悉命令行 2、已安装 16.0 或更高版本的 Node.js 参照vuejs官网的步骤&#xff0c;创建一个vue前端项目 当前vuejs的版本&#xff1a;3.2.47 npm init vuelatestVue.js - The Progressive JavaScript Framework√ Project name: ... vuejs3-project√ Add Type…

BitDock桌面美化工具 一直在后台偷偷上传东西,具体上传什么东西不知,一天耗费我几十个G的流量

通过流量防火墙监控发现bitdock一直在上传东西&#xff0c;目前截止发现已上传了40G的流量 ――――――――――――――――――――――― 程序名称&#xff1a;SystemAudioDetection.exe 程序说明&#xff1a; 路径&#xff1a;D:\BitDock\AudioEngine\SystemAudioDetecti…

【C 字符串】02 字符串函数(命令行参数)

Navigator一、strlen()函数—统计长度二、strcat()函数—拼接三、strncat()函数—strcat()的升级四、strcmp()和strncmp()—比较五、strcpy()和strncpy()—拷贝六、sprintf()函数—合并多个字符串七、其他可能用到的字符串函数八、ctype.h中的字符函数九、把字符串转换为数字十…

在线文章生成工具-原创文章生成工具

在线文章生成器 在线文章生成器是指一种可以在线使用的自动化创造文章的工具。它可以使用自然语言处理&#xff08;NLP&#xff09;技术和人工智能算法提供需要的信息&#xff0c;基于标题、关键字&#xff0c;句子关联性等元素自动创造文章内容&#xff0c;涵盖各种类型&…

双端队列 码蹄集

题目来源&#xff1a;码蹄集 题目描述&#xff1a; 题意分析&#xff1a; 这道题目需要使用到双端队列的数据结构。我们可以借助 STL 中的 deque 来实现这个数据结构。具体来说&#xff0c;我们可以通过 deque 的 push_front 和 push_back 操作在队列的头部和尾部添加元素&am…

地球系统模式(CESM)实践技术应用

目前通用地球系统模式&#xff08;Community Earth System Model&#xff0c;CESM&#xff09;在研究地球的过去、现在和未来的气候状况中具有越来越普遍的应用。于2010年07月推出以来&#xff0c;一直受到气候学界的密切关注。近年升级的CESM2.0在大气、陆地、海洋、海冰、陆冰…

图形库EasyX的学习:)

最近学了一点做小游戏的基本知识&#xff0c;来总结一下&#xff0c;巩固一下记忆&#xff1a; 在这个基础上初学者要先明白vs的下载及基本使用还有图形库的下载及安装&#xff1b; 然后才是正题&#xff1a; 图形库里包含c语法&#xff0c;所以要用c文件&#xff0c;但是除…

Ubantu docker学习笔记(六)容器数据卷——补充实验

文章目录一、volume container二、 data-packed volume container三、利用数据卷驱动共享数据注意要在同一个网络配置下&#xff01;3.1服务端3.2客户端一、volume container 这里我觉得很好理解&#xff0c;volume container是专门为其他容器提供volume的容器。其实也就相当于…

深圳海运到墨西哥需要多长时间

目前&#xff0c;墨西哥的跨境电商商业正在高速发展&#xff0c;并且具有可观的红利。因此&#xff0c;从中国到墨西哥的运输需求很大&#xff0c;特别是海运&#xff0c;是很多跨境电商卖家主要选择的运输方式。 一般而言&#xff0c;中国到墨西哥的跨境卖家们普遍关注海运所需…

“终于我从字节离职了...“一个年薪40W的测试工程师的自白...

”我递上了我的辞职信&#xff0c;不是因为公司给的不多&#xff0c;也不是因为公司待我不好&#xff0c;但是我觉得&#xff0c;我每天看中我憔悴的面容&#xff0c;每天晚上拖着疲惫的身体躺在床上&#xff0c;我都不知道人生的意义&#xff0c;是赚钱吗&#xff1f;是为了更…

【云原生进阶之容器】第五章容器运行时5.8--容器热迁移

《云原生进阶之容器》专题索引: 第一章Docker核心技术1.1节——Docker综述第一章Docker核心技术1.2节——Linux容器LXC第一章Docker核心技术1.3节——命名空间Namespace第一章Docker核心技术1.4节——chroot技术第一章Docker核心技术1.5.1节——cgroup综述

MySQL-存储过程

什么是存储过程我们前面所学习的MySQL语句都是针对一个表或几个表的单条 SQL 语句&#xff0c;但是在数据库的实际操作中&#xff0c;并非所有操作都那么简单&#xff0c;经常会有一个完整的操作需要多条SQL语句处理多个表才能完成。例如&#xff0c;为了确认学生能否毕业&…

015 - C++ 类与结构体对比

今天这期我们主要解决一个问题&#xff0c;就是 C 中的类和结构体有什么区别。 上一期我们讲类的时候&#xff0c; 我们对类有了一些基本的介绍&#xff0c;在本期的学习开始之前你可以先看看那一期。 本期我们有两个术语&#xff0c;结构体 struct&#xff0c;它是 structur…