Linux | Linux使用互斥锁及条件变量替代信号量

2023-11-05 02:40

本文主要是介绍Linux | Linux使用互斥锁及条件变量替代信号量,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 一、简述
    • 二、测试

========》Linux | Linux中的线程、互斥量、信号量的基本使用《========

一、简述

信号量是一个计数器,用于限制并发访问共享资源的线程数;

  • 当计数器严格大于0时,对Wait()的调用立即返回并递减计数器;
  • 为0时,对Wait的任何后续调用都会阻塞,并且仅在信号量计数器再次变为严格正数时返回,调用Post()会增加计数器;
  • sem(1)即二元信号量相当于mutex,区别在于,mutex只能由获取锁的线程释放,而信号量属于线程中通信的机制,wait和post可以在不同线程见调用故容易出现问题;

【应用场景】

  • 通常,信号量可用于限制对共享资源的访问,该共享资源只能由某些固定数量的客户端同时访问;
    • 例如,在对酒店预订系统建模时,可以创建计数器等于可用房间总数的信号量。每次预留房间时,应通过调用Wait()获取信号量,每次释放房间时应通过调用Post释放信号量;
C++11 和 Boost.Thread 都没有提供信号量:信号量因太容易出错而被删除。通过互斥体和条件变量的组合可以更安全地实现相同的效果;Dijkstra(信号量的发明者)、Hoare 和 Brinch Hansen 都贬低了信号量并提倡更结构化的替代方案。在 1969 年给 Brinch Hansen 的一封信中,Wirth 说“信号量……不适合高级语言。” [Andrews-83] 将典型错误总结为“省略P或V ,或在一个信号量上不小心将P和V编码在另一个信号量上”,忘记在临界区中包含对共享对象的所有引用,以及由于使用“条件同步和互斥”的相同原语;

【p】故以下我们将使用条件变量和互斥量来替代信号量

/** 封装条件变量 */
class Cond {
public:Cond();void wait(pthread_mutex_t mutex);void signal();~Cond();private:pthread_cond_t m_cond;
};/** 信号量封装使用条件变量和锁 */
class OwnSemaphore : Noncopyable {
public:typedef Mutex MutexType;OwnSemaphore(size_t count=0);~OwnSemaphore();void wait();void notify();size_t getCount() const { return m_count; }void reset() { m_count = 0;}private:size_t m_count;MutexType m_mutex;Cond m_cond;
};OwnSemaphore::OwnSemaphore(size_t count):m_count(count){
}OwnSemaphore::~OwnSemaphore() {
}void OwnSemaphore::wait() {MutexType::Lock lock(m_mutex);while (m_count == 0) {m_cond.wait(m_mutex.getMutex());}--m_count;
}void OwnSemaphore::notify() {m_count++;m_cond.signal();
}

二、测试

每个工作线程先等待信号量,然后输出线程 ID 和当前时间,输出操作以互斥锁同步以防止错位,睡眠一秒是为了模拟线程处理数
据的耗时。
sylar::OwnSemaphore g_sem(3);void Worker() {g_sem.wait();std::thread::id thread_id = std::this_thread::get_id();/* 获取时间 */struct tm tm;time_t t = time(0);localtime_r(&t, &tm);char buf[64];strftime(buf, sizeof(buf), "%H:%M:%S", &tm);{sylar::Mutex::Lock lock(sylar::Mutex);std::cout << "Thread " << thread_id << ": wait succeeded" << " (" << buf << ")" << std::endl;}// Sleep 1 second to simulate data processing.std::this_thread::sleep_for(std::chrono::seconds(1));g_sem.notify();
}int main() {const std::size_t SIZE = 3;std::vector<std::thread> v;v.reserve(SIZE);for (std::size_t i = 0; i < SIZE; ++i) {v.emplace_back(&Worker);}for (std::thread& t : v) {t.join();}return 0;
}

测试结果
在这里插入图片描述
在这里插入图片描述

参考文章
https://segmentfault.com/a/1190000006818772

这篇关于Linux | Linux使用互斥锁及条件变量替代信号量的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/346864

相关文章

防止Linux rm命令误操作的多场景防护方案与实践

《防止Linuxrm命令误操作的多场景防护方案与实践》在Linux系统中,rm命令是删除文件和目录的高效工具,但一旦误操作,如执行rm-rf/或rm-rf/*,极易导致系统数据灾难,本文针对不同场景... 目录引言理解 rm 命令及误操作风险rm 命令基础常见误操作案例防护方案使用 rm编程 别名及安全删除

Linux下MySQL数据库定时备份脚本与Crontab配置教学

《Linux下MySQL数据库定时备份脚本与Crontab配置教学》在生产环境中,数据库是核心资产之一,定期备份数据库可以有效防止意外数据丢失,本文将分享一份MySQL定时备份脚本,并讲解如何通过cr... 目录备份脚本详解脚本功能说明授权与可执行权限使用 Crontab 定时执行编辑 Crontab添加定

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

C#使用Spire.Doc for .NET实现HTML转Word的高效方案

《C#使用Spire.Docfor.NET实现HTML转Word的高效方案》在Web开发中,HTML内容的生成与处理是高频需求,然而,当用户需要将HTML页面或动态生成的HTML字符串转换为Wor... 目录引言一、html转Word的典型场景与挑战二、用 Spire.Doc 实现 HTML 转 Word1

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

MyBatis ParameterHandler的具体使用

《MyBatisParameterHandler的具体使用》本文主要介绍了MyBatisParameterHandler的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录一、概述二、源码1 关键属性2.setParameters3.TypeHandler1.TypeHa

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完

使用docker搭建嵌入式Linux开发环境

《使用docker搭建嵌入式Linux开发环境》本文主要介绍了使用docker搭建嵌入式Linux开发环境,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录1、前言2、安装docker3、编写容器管理脚本4、创建容器1、前言在日常开发全志、rk等不同