TinyWebServer学习笔记-互斥锁、信号量、条件变量

为什么要使用锁、信号量、条件变量?

网站不可能是单线程的,否则网站的性能和响应都会收到严重的影响。因此,这个项目一定是运行在多线程条件下的。而在多线程条件下,对共享资源的互斥访问就极其重要。

为什么要将资源封装成类?

首先,我们要明确资源的使用一般有三个步骤:

1、获取资源

2、使用资源

3、释放资源

然而,大多数情况下,我们一定能做到前两点,而总是忘记第三步,这就会造成资源的泄露。为了解决这个问题,提出了RAII方案,中文翻译是资源获取即初始化,也就是使用局部对象来管理资源的技术。

最直观的就是将资源封装成类, 在构造函数中获取资源,在析构函数中释放资源,这样当变量离开了作用域后,编译器会调用析构函数,而析构函数会帮助我们将资源释放掉,这样就避免了我们忘记释放资源的情况。

本项目中,作者就将锁、互斥量、条件变量进行了封装。有以下好处:

  1. 自动资源管理:RAII模式允许资源的获取和释放与对象的生命周期绑定在一起。当对象被创建时,资源被获取;当对象被销毁时,资源被释放。这消除了手动管理资源的需要,减少了错误的机会。

  2. 异常安全性:RAII模式提供了异常安全性,因为资源的释放操作通常在对象的析构函数中执行。如果在使用资源的过程中发生异常,对象的析构函数会被自动调用,确保资源被正确释放,防止资源泄漏。

  3. 避免忘记释放资源:使用RAII模式,开发人员不需要显式地记住在何处释放资源。资源的释放是自动的,因此避免了因忘记释放资源而引发的问题。

  4. 代码可读性:RAII模式使代码更加清晰和容易理解。资源的获取和释放操作都在对象的构造函数和析构函数中,使得代码更加自文档化。

  5. 资源的精确生命周期控制:RAII模式允许在对象的生命周期内对资源的生命周期进行精确的控制。资源在对象的构造和析构之间一直存在,不会在对象的其他方法之外被访问。

  6. 并发安全性:RAII模式可以用于管理锁、信号量等同步资源,确保线程安全性。资源在锁住和释放锁时被获取和释放,从而避免了竞态条件和死锁等问题。

 互斥锁、条件变量、信号量是如何工作的?

互斥锁:提供互斥访问的能力来确保同一时间只有一个线程能够访问共享资源。

        1. 锁初始化:创建锁并进行初始化;

        2. 加锁:当线程需要访问共享资源时,尝试获取互斥锁,如果已经被别的线程占用,那么该线程会被阻塞,直到锁可用;

        3. 访问共享资源:获取锁后,线程访问共享资源;

        4. 解锁:该线程访问完共享资源后,释放互斥锁,使其他等待的线程可以获得锁并访问资源。

条件变量:通常和互斥锁一起使用,条件变量用于等待某个条件满足后再继续执行。

        1. 等待条件:线程获取互斥锁后,发现条件不足,进入阻塞状态,释放互斥锁,让其他进程可以访问共享资源;

        2. 唤醒等待进程:当条件满足后,某个线程通过条件变量发出信号,唤醒一个或多个等待的线程,这些线程被唤醒后后会尝试重新获取互斥锁。

信号量:用于控制多个线程对有限资源的访问。通常有两种:

        1.二进制信号量:非0即1,用于互斥访问共享资源。等待信号量为1时获得访问权,将信号量设置为0,访问结束后将信号量值设为1;

        2. 计数信号量:可以具有>1的值,控制多个线程对一组有限资源的访问。线程获得资源信号量值变小;释放资源信号量值变大,当信号量值为0时需要阻塞线程。

下面我们来看作者的源代码:

#ifndef LOCKER_H
#define LOCKER_H#include <exception>
#include <pthread.h>
#include <semaphore.h>class sem
{
public:sem(){if (sem_init(&m_sem, 0, 0) != 0){throw std::exception();}}sem(int num){if (sem_init(&m_sem, 0, num) != 0){throw std::exception();}}~sem(){sem_destroy(&m_sem);}bool wait(){return sem_wait(&m_sem) == 0;}bool post(){return sem_post(&m_sem) == 0;}private:sem_t m_sem;
};
class locker
{
public:locker(){if (pthread_mutex_init(&m_mutex, NULL) != 0){throw std::exception();}}~locker(){pthread_mutex_destroy(&m_mutex);}bool lock(){return pthread_mutex_lock(&m_mutex) == 0;}bool unlock(){return pthread_mutex_unlock(&m_mutex) == 0;}pthread_mutex_t *get(){return &m_mutex;}private:pthread_mutex_t m_mutex;
};
class cond
{
public:cond(){if (pthread_cond_init(&m_cond, NULL) != 0){//pthread_mutex_destroy(&m_mutex);throw std::exception();}}~cond(){pthread_cond_destroy(&m_cond);}bool wait(pthread_mutex_t *m_mutex){int ret = 0;//pthread_mutex_lock(&m_mutex);ret = pthread_cond_wait(&m_cond, m_mutex);//pthread_mutex_unlock(&m_mutex);return ret == 0;}bool timewait(pthread_mutex_t *m_mutex, struct timespec t){int ret = 0;//pthread_mutex_lock(&m_mutex);ret = pthread_cond_timedwait(&m_cond, m_mutex, &t);//pthread_mutex_unlock(&m_mutex);return ret == 0;}bool signal(){return pthread_cond_signal(&m_cond) == 0;}bool broadcast(){return pthread_cond_broadcast(&m_cond) == 0;}private://static pthread_mutex_t m_mutex;pthread_cond_t m_cond;
};
#endif

互斥锁和信号量是简单的封装,而对于条件变量作者注释掉了互斥锁,但并不会影响互斥访问,因为作者选择用封装好的互斥锁并且在外部来使用。例如在block_queue.h这个文件中,作者将locker m_mutex;作为私有数据成员,并通过get函数获得该锁。

template <class T>
class block_queue
{
public:...if (!m_cond.timewait(m_mutex.get(), t)){m_mutex.unlock();return false;}...
private:locker m_mutex;...
}

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

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

相关文章

IDEA中的神仙插件——Smart Input (自动切换输入法)

推荐专栏&#xff1a;开发环境配置攻略 致力于记录学习过程中各种软件的安装及环境配置操作&#xff0c;并提供详细的步骤说明和丰富的配图。涵盖了 Java、Python、IntelliJ IDEA、Tomcat、MySQL 等常见开发工具和服务器组件的配置&#xff0c;为初学者提供一个实用、全面的配置…

Pandas进阶修炼120题-第五期(一些补充,101-120题)

目录 往期内容&#xff1a;第一期&#xff1a;Pandas基础&#xff08;1-20题&#xff09;第二期&#xff1a;Pandas数据处理&#xff08;21-50题&#xff09;第三期&#xff1a;Pandas金融数据处理&#xff08;51-80题&#xff09;第四期&#xff1a;当Pandas遇上NumPy&#xf…

Vue2 第一次学习

本章为超级浓缩版,文章过于短,方便复习使用哦~ 文章目录 1. 简单引入 vue.js2. 指令2.1 事件绑定指令 v-on (简写 )2.2 内容渲染指令2.3 双向绑定指令 v-model2.4 属性绑定指令 v-bind (简写 : )2.5 条件渲染指令2.6 循环指令 v-for 3. vue 其他知识3.1 侦听器 watch3.2 计算属…

WOL唤醒配置(以太网、PHY、MAC)

目录 wol 以太网 MAC PHY RMII 通信配置 总结 wol Wake-on-LAN简称WOL&#xff0c;WOL&#xff08;网络唤醒&#xff09; 是一种标准网络协议&#xff0c;它的功效在于让已经进入休眠状态或关机状态的计算机&#xff0c;透过局域网&#xff08;多半为以太网&#xff…

lv8 嵌入式开发-网络编程开发 01什么是互联网

目录 1 计算机网络的定义与分类 1.1 按照网络的作用范围进行分类 1.2 按照网络的使用者进行分类 2 网络的网络 2.1 名词解释 2.2 边缘与核心 3 互联网基础结构发展的三个阶段 3.1 第一阶段&#xff1a;1969 – 1990 3.2 第二阶段&#xff1a;1985 – 1993 3.3 第三阶…

HTML5高级部分

目录 一、拖拽API1.1 拖拽元素1.2 监听事件1.3 dataTransfer传递数据 二、媒体API2.1 常用监听事件2.2 常用API 三、画布API3.1 canvas 标签3.2 创建canvas对象3.3 常用API 四、地理API4.1 方法 一、拖拽API 1.1 拖拽元素 页面中设置了draggable"true"的元素可以进…

【Java-LangChain:使用 ChatGPT API 搭建系统-11】用 ChatGPT API 构建系统 总结篇

第十一章&#xff0c;用 ChatGPT API 构建系统 总结篇 本课程详细介绍了 LLM 工作原理&#xff0c;包括分词器&#xff08;tokenizer&#xff09;的细节、评估用户输入的质量和安全性的方法、使用思维链作为 Prompt、通过链式 Prompt 分割任务以及返回用户前检查输出等。 本课…

linux MySQL高阶语句

linux MySQL高阶语句 1、MySQL高级语言1.1order by排序1.2group by分组1.3limit前几行1.4as别名1.5通配符1.6子查询1.7in1.8not in1.9exists 2、视图2.1视图概念2.2功能2.3应用场景2.4视图和表的区别和联系2.5创建视图 3、null值3.1null值3.2null值与空值的区别3.3验证null和空…

僵尸进程的产生原因和解决方法

僵尸进程的产生原因 当一个进程&#xff08;通常是父进程&#xff09;创建了一个子进程&#xff0c;但是在子进程终止后&#xff0c;父进程没有及时处理子进程的终止状态&#xff0c;就会导致僵尸进程的产生。这个时候&#xff0c;子进程虽然已经终止&#xff0c;但是其进程表…

Autowired和Resource的关系

相同点对于下面的代码来说&#xff0c;如果是Spring容器的话&#xff0c;两个注解的功能基本是等价的&#xff0c;他们都可以将bean注入到对应的field中 不同点但是请注意&#xff0c;这里说的是基本相同&#xff0c;说明还是有一些不同点的&#xff1a; byName和byType匹配顺…

IDEA的Maven换源

前言 IDEA是个好东西&#xff0c;但是使用maven项目时可能会让人很难受&#xff0c;要么是非常慢&#xff0c;要么直接下载不了。所以我们需要给IDEA自带maven换源&#xff0c;保证我们的下载速度。 具体操作 打开IDEA安装路径&#xff0c;然后打开下面的文件夹 plugins\m…

Armv8/9-A cpu在安全/非安全世界切换时,是否需要对共享内存进行cache维护操作?

安全之安全(security)博客目录导读 问题&#xff1a;当Armv8/9-A cpu在安全世界和非安全世界之间切换时&#xff0c;是否需要对这两个世界的共享内存进行缓存维护操作? 答案&#xff1a; 不需要。 1&#xff09;运行在非安全世界的软件只能对内存进行非安全访问&#xff0c…

第10章 MySQL(一)

10.1 谈谈MySQL的架构 难度:★★ 重点:★ 白话解析 要想彻底的理解MySQL,它的架构一定要先弄清楚,当Java程序员通过JDBC或者Mybatis去执行一条SQL的时候,到底经历了什么。下边先看一幅图: 户端:Java程序员通过JDBC或者Mybatis去拿MySQL的驱动程序,实际上就是拿客户端。…

java基础之构造器

构造器 学习java对于构造器应该很熟悉&#xff0c;但是有些人会认为构造器不是必要的&#xff0c;这就是对于构造器没有深入的了解。 每一个java类中都必须至少有一个显式或隐式的构造器&#xff0c;很多时候看到类中并没有定义构造器&#xff0c;有人会认为构造器不是必须的&a…

GNN PyG~torch_geometric 学习理解

目录 1. PyG Introduction 2. PyG Installation 2.1 PyG 安装常见错误及原因 2.2 PyG 具体安装步骤 3. torch_geometric packages torch_geometric.data.Data Dataset 与 DataLoader Dropout、BatchNorm 3. torch_geometric: 理解edge_index 3.1 理解 mini-batch edg…

【Java】SpringMVC ResponseBodyAdvice详解

目录 1. ResponseBodyAdvice 2. supports方法 3. beforeBodyWrite方法 4. 实践 1. ResponseBodyAdvice Spring MVC的ResponseBodyAdvice是Spring 4.1版本中引入的一个接口&#xff0c;它允许在Controller控制器中ResponseBody修饰的方法或ResponseEntity执行之后&#xff…

【2023年11月第四版教材】第17章《干系人管理》(合集篇)

第17章《干系人管理》&#xff08;合集篇&#xff09; 1 章节内容2 管理基础3 管理过程3.1 管理的过程★★★ &#xff08;22上44&#xff09;3.2 管理ITTO汇总★★★ 4 过程1-识别干系人4.1 数据收集★★★4.3数据分析4.4 权力利益方格4.5 数据表现&#xff1a;干系人映射分析…

记录UNIAPP打包苹果iOS·APP

用到生成的四个文件:1-1.CSR证书文件、2-2.CER证书文件、3-3.PP文件【证书Profiles文件】、4-4.P12文件【证书私钥】 1. 生成CSR证书文件: 2. 操作苹果后台:Sign In - Applehttps://developer.apple.com/account/resources/certificates/list

使用CrawlSpider爬取全站数据。

CrawpSpider和Spider的区别 CrawlSpider使用基于规则的方式来定义如何跟踪链接和提取数据。它支持定义规则来自动跟踪链接&#xff0c;并可以根据链接的特征来确定如何爬取和提取数据。CrawlSpider可以对多个页面进行同样的操作&#xff0c;所以可以爬取全站的数据。CrawlSpid…

PHP图片文件管理功能系统源码

文件图库管理单PHP源码直接解压就能用&#xff0c;单文件&#xff0c;indexm.php文件可以重新命名&#xff0c;上传到需要访问的目录中&#xff0c; 可以查看目录以及各个文件&#xff0c;图片等和下载及修改管理服务。 源码下载&#xff1a;https://download.csdn.net/downloa…