【Linux】多线程 之 POSIX信号量

news2025/1/11 17:12:15

文章目录

  • 1. 概念
  • 2. 信号量的工作机制
  • 3. 认识接口
    • sem_init ——初始化信号量
    • sem_destroy——销毁信号量
    • sem_wait ——申请信号量
    • sem_post ——释放信号量
  • 4. 基于环形队列的生产消费模型
    • 原理解析
    • 代码
      • 代码解析
        • ringqueue类
          • 构造
          • 析构
          • push ——生产
          • pop ——消费
      • 代码实现
        • Ringqueue.hpp
        • makefile
        • main.cc

1. 概念

信号量又称为 信号灯
本质就是一个计数器,用于描述临界资源数目的

sem: 0 -> 1 -> 0
若临界资源只有1个,则sem设为1,当要使用临界资源时,sem由1变为0,其他人在想申请,则申请不到挂起排队,等待释放临界资源时 sem由0变为1 ,才可以再申请临界资源
这种信号量称为 二元信号量 ,等同于互斥锁

每一个线程,在访问对应的资源时,先申请信号量,
申请成功,表示该线程允许使用该资源
申请不成功,表示目前无法使用该资源

2. 信号量的工作机制

信号量机制类似于看电影买票,一种资源的预订机制
申请信号量成功,相当于预定了一部分资源

判断条件是否满足,决定了后续行为
信号量已经是资源的计数器,申请信号量成功,本身就表明资源可用
申请信号量失败,本身表明资源不可用
本质就是把判断转换成信号量的申请行为

3. 认识接口

POSIX信号量 和system V 信号量 作用相同,都是用于同步操作,达到无冲突的访问共享资源目的,但POSIX可以用于线程间同步


sem_init ——初始化信号量

输入 man sem_init

sem :表示信号量
pshared : 0表示线程间共享 非零表示进程间共享
value : 信号量初始值 (计数器值初始化为多少)

sem_destroy——销毁信号量

输入 man sem_destroy

对已经初始化的信号量进行销毁

sem_wait ——申请信号量

输入 man sem_wait

进行申请信号量的操作,使信号量的值减1

sem_post ——释放信号量

输入 man sem_post

进行释放信号量的操作,使信号量的值加1

4. 基于环形队列的生产消费模型

原理解析

环形队列实际上使用数组模拟的

数组多开一个空间是为了解决判满的问题


若为空,则 thread和tail 在同一个位置


若为满,则tail的下一个位置为head


生产者向tail中push数据 即生产
消费者向head中pop数据 即消费


生产者 和消费者 关心的资源 是一样的吗?
不一样, 生产者关心整个环形队列的空间(商店是否装满货物)
消费者关心的是 数据,(商店是否还有货物,有货物就买)


head 和tail什么时候访问 同一个区域?
有一个很大的桌子,存在像钟表的0-12点刻度的区域,在每个刻度中放入一个盘子
有两个人A和B同时进入房间看到桌子,A往盘子中放苹果,B在后面拿苹果
A和B约定:B不能超过A,一个盘子只能放一个苹果

当A和B开始,桌子上没有苹果时 ,或者 桌子上全都是苹果时,都会访问同一个盘子
环形队列 为空 ,或者环形队列为满 会访问 同一个区域

当队列为空,指向同一个位置,存在竞争关系,
让生产者先运行 (只有当生产者产生数据后,消费者才能拿到数据)

当队列为满时,指向同一个位置,存在竞争关系,
让消费者先运行 (只有当消费者拿数据后,生产者才能生产)


生产者关心空间,空间本身也是资源,所以要给生产者定义一个信号量sem_room ,其初始值为N
P(sem_room) —— 申请空间信号量

生产者生产数据在当前空间,则对应的数据+1,所以消费者可以拿数据
V(sem_data) ——数据信号量的值+1

消费者关心数据,信号量为sem_data,其初始值为0
P(sem_data) —— 申请数据信号量

消费者把数据拿走,当前空间就被闲置出来了,所以生产者可以放数据
V(sem_room) ——空间信号量的值+1

代码

代码解析

首先在ringqueue.hpp中创建一个ringqueue类


在main函数中使用new创建出rq队列
为了保证生产者和消费者看到同一份资源,所以两者回调函数的参数args都为rq


productorRoutine的回调函数中 使用 队列rq的push,将数据插入到队列中 即生产
consumerRoutine的回调函数中 使用 队列rq的pop,把队列中的数据取出 即消费


ringqueue类

ringqueue类中

在上述讲解原理时,数据信号量只有消费者关心,空间信号量只有生产者关心

构造

将环形队列ring大小和_cap(容量)初始化为N

0表示线程间共享,将数据信号量 初始化为0,将空间信号量初始化为整个环形队列的容量
(对于两者的初始化值大小,在原理处都有详细解释)

析构

由于在构造时,对信号量进行初始化,所以需要销毁信号量

push ——生产

要生产之前要保证符合条件,才能够进行生产,所以要进行P操作——申请信号量

在使用信号量时,是不需要判断的
因为信号量是一把计数器,本质为把对资源就绪的情况,由在临界区内转到临界区外
它本身就是描述临界资源数量的,所以就不用进入临界区后判断临界资源是否满足条件


生产者和消费者可能访问同一个位置,大概率访问不同的位置
所以生产者和消费者要有自己的下标 用于 表示两者的位置


不断进行P操作,在空间上插入数据
没有空间就需要消费者进行消费(V操作),将数据拿走下·

当tail达到多开一个空间位置,实际上相当于再次回到head开头的位置
所以使用%=,模拟环形队列


将 sem_wait 和sem_post借助 函数 P和V完成封装
再次使用时,只需调用P V即可实现

在这里插入图片描述

pop ——消费

不断进行P操作,将数据从空间上拿走,空间都闲置出来了
就需要生产者进行生产(V操作),在空间上放置数据

代码实现

Ringqueue.hpp


#include<iostream>
#include<vector>
#include<semaphore.h>//信号量头文件
static const int N=5;//设置环形队列的大小

template<class T>
class ringqueue
{
private:
  void P(sem_t &s)
  {
     sem_wait(&s); 
  }
  void V(sem_t&s)
  {
     sem_post(&s);
  }
public:
    ringqueue(int num=N)
    : _ring(num),_cap(num)
    {
        //信号量初始化
        sem_init(&_data_sem,0,0);
        sem_init(&_space_sem,0,num);
        _c_step=_p_step=0;//生产和消费下标都为0
    }
    ~ringqueue()
    {
        //销毁信号量
     sem_destroy(&_data_sem);
     sem_destroy(&_space_sem);
    }

    void push(const T&in)//生产
    {
      P(_space_sem);//P操作 申请信号量
      _ring[_p_step++]=in;//将数据放入生产位置
      _p_step%=_cap;
      V(_data_sem);//V操作 释放信号量
    }
    void pop(T*out)//消费 
    {
       P(_data_sem);//P操作
       *out=_ring[_c_step++];//将该位置的数据给与out
       _c_step%=_cap;
       V(_space_sem);//V操作
    }
private:
   int _c_step;//消费者位置下标
   int _p_step;//生产者位置下标
   std::vector<int> _ring;//充当环形队列
   int  _cap;//环形队列的容器大小
   sem_t _data_sem;//数据信号量
   sem_t _space_sem;//空间信号量 
};

makefile

ringqueue:main.cc
	g++ -o $@ $^ -std=c++11 -lpthread
.PHONY:clean
clean:
	rm -f ringqueue 

main.cc


#include"Ringqueue.hpp"
#include<pthread.h>
#include<unistd.h>
using namespace std;
void*consumerRoutine(void*args)
{
  ringqueue<int>*rq=(ringqueue<int>*)args;
  while(true)
  {
  int data=0;
  rq->pop(&data);//从队列取出数据 消费
  cout<<"consumer done:"<<data<<endl;
  sleep(1);
  }
}
void*productorRoutine(void*args) 
{
 ringqueue<int>*rq=(ringqueue<int>*)args;
 while(true)
 {
   int data=1;
   rq->push(data);//将数据插入队列中 生产
   cout<<"productor done:"<<data<<endl;
 }
}

int main()
{
   ringqueue<int>*rq=new ringqueue<int>();
   pthread_t c;//消费者
   pthread_t p;//生产者
   //创建线程
   pthread_create(&c,nullptr,consumerRoutine,rq);
     pthread_create(&p,nullptr,productorRoutine,rq);

     pthread_join(c,nullptr);
     pthread_join(p,nullptr);
    return 0;
}

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

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

相关文章

精通Skia引擎,发挥应用程序中高性能图形的潜力

Skia是由Google开发的一个开源的2D图形库。它提供了丰富的图形绘制、图像处理和渲染功能&#xff0c;适用于多个平台&#xff0c;包括操作系统、浏览器和移动设备等。Skia主要用于处理2D图形的绘制、文字渲染、图像处理和用户界面的呈现。 特点和功能&#xff1a; 高性能&…

基于Javaweb实现ATM机系统开发实战(十一)存储交易记录

首先创建一个业务接口&#xff1a; package com.atm.service;import com.atm.pojo.RunMessage;//交易记录的业务接口 public interface RunMessageService{//添加交易记录public void addRunMessage(RunMessage runMessage) throws Exception ; }再完成业务接口的实现类&#…

iNav开源代码之FAILSAFE

iNav开源代码之FAILSAFE 1. 源由2. FAILSAFE类别3. FAILSAFE配置4. FAILSAFE阶段&状态机5. 参考资料 1. 源由 最近因为炸机&#xff0c;百思不得其解。 关于炸鸡的过程&#xff0c;就不再展开&#xff0c;都是“泪”啊&#xff01;想进一步了解的&#xff0c;请参阅前面的…

科普一下Elasticsearch中BM25算法的使用

首先还是先了解几个概念&#xff0c;Elasticsearch是一个开源的分布式搜索和分析引擎&#xff0c;它使用一系列算法来计算文档的相关性分数&#xff08;relevance score&#xff09;。这些算法用于确定查询与文档的匹配程度&#xff0c;以便按相关性对搜索结果进行排序。以下是…

【C++】多线程编程一(初识并发和多线程)

目录 一、并发和并行的概念 1.并发 2.并行 3.并发和并行的区别 二、并发的途径 多进程并发 多线程并发 三、C11相关多线程编程的头文件 一、并发和并行的概念 1.并发 并发&#xff1a;指同一时刻只能有一条指令执行&#xff0c;但是多个进程指令被快速地轮换执行&#…

C语言--文件操作

文章目录 前言文件文件名 文件的打开和关闭文件指针文件的打开和关闭 文件的顺序读写fgetc、fputcfgets、fputsfprintf、fsanffread、fwrite 文件的随机读写fseekftellrewind 文本文件和二进制文件文件读取结束的判定feof 文件缓冲区 前言 在我们写完一个程序之后&#xff0c;…

【数据结构】树二叉树的概念以及堆的详解

⭐️ 往期相关文章 ✨链接1&#xff1a;【数据结构】顺序表 ✨链接2&#xff1a;【数据结构】单链表 ✨链接3&#xff1a;【数据结构】双向带头循环链表 ✨链接4&#xff1a;【数据结构】栈和队列 ⭐️ 树的概念 百度百科的解释&#xff1a;树是一种非线性的数据结构&#xf…

Jmeter四种关联方法讲解

目录 方法一&#xff0c;从前一个请求中取&#xff0c;用正则表达式提取器。 二、json path extractor 三、json extractor 四、XPath Extractor 方法一&#xff0c;从前一个请求中取&#xff0c;用正则表达式提取器。 具体方法&#xff0c;在需要获得数据的请求上右击添加…

【09】STM32·HAL库-新建HAL库版本MDK工程 | 下载STM32固件库

目录 1&#xff0c;新建工程前的准备工作&#xff08;了解&#xff09;1.1下载相关STM32Cube 官方固件包&#xff08;F1/F4/F7/H7&#xff09; 2&#xff0c;新建HAL库版本MDK工程步骤&#xff08;熟悉&#xff09;2.1新建工程文件夹2.1.1Drivers文件夹2.1.2Middlewares文件夹2…

基数排序(Radix_Sort)

基数排序 (Radix Sort)-20230715 前言 基数排序适用于多关键字排序&#xff0c;与前述的比较排序不同&#xff0c;实现基数排序不需要对关键字进行比较和移动。简而言之&#xff0c;基数排序是一类借助多关键字排序的思想对单逻辑关键字实现排序的方法。 多关键字排序 先看…

SwiftUI 原生或利用 Vision 检测限定高度的 Text 视图能否完整显示文本的方法

功能需求 在 SwiftUI 开发中,为了节省空间我们往往会为内容很长的文本视图(Text)限定一个高度,然后让用户决定是否展开显示其完整内容。 如上图所示,为了节省空间我们对 Text 视图高度做了限制,然后根据文本长度自动显示或隐藏展开按钮,用户点击该按钮即可展开显示完整…

FL Studio21入门版编曲 2023年免费小白新手编曲工具

全能数字音乐工作站&#xff08;DAW&#xff09;编曲、剪辑、录音、混音&#xff0c;有了它就能把你的笔记本电脑变成全功能音乐工作室。 内置丰富插件&#xff0c;满足不同风格创作拥有强大的采样引擎&#xff0c;自带高品质打击乐、钢琴、弦乐、吉他等107种乐器效果。 流行…

C++-把字符串转换成整数

题目来源&#xff1a;牛客网 题目描述&#xff1a; 将一个字符串转换成一个整数&#xff0c;要求不能使用字符串转换整数的库函数。 数值为 0 或者字符串不是一个合法的数值则返回 0 数据范围&#xff1a;字符串长度满足 0≤n≤100 进阶&#xff1a;空间复杂度 O(1) O(1) &…

Mybatis执行SQL过程

文章目录 1. 相关代码2. 创建SqlSession3. 创建Mapper代理对象4.sql的执行4.1 MapperProxy.invoke()4.2 mapperMethod.execute()4.3 sqlSession.selectOne4.4 CachingExecutor.query()4.5 BaseExecutor.query方法4.6 SimpleExecutor.doQuery方法 1. 相关代码 Testpublic void …

其他形式转欧拉角形式

1. 坐标系轴方向问题 3D数学基础中约定使用左手坐标系 左手坐标系 右手坐标系 左手正方向&#xff1a;x正向右平移&#xff0c;y向上平移&#xff0c;z向前平移. 右手正方向&#xff1a;x正向左平移&#xff0…

漫谈大数据时代的个人信息安全(二)——“逢脸造戏”

大数据时代的个人信息安全系列二&#xff1a;“逢脸造戏” 1. 逢脸造戏2. 生物识别信息安全2.1 生物识别信息被大量获取2.2 生物识别信息被非法滥用 3. 各国加强对深度合成监管4. 个人信息保护小贴士 互联网就像公路&#xff0c;用户使用它&#xff0c;就会留下脚印。 每个人都…

文件IO_复制文件描述符(附Linux-5.15.10内核源码分析)

目录 1.文件描述符复制简介 2.dup函数原型 2.1 dup函数 2.2 dup函数工作原理 2.3 dup函数内核源码分析 2.4 dup函数示例代码 3.dup2函数原型 3.1 dup2函数 3.2 dup2函数工作原理 3.3 dup2函数内核源码分析 3.4 dup2函数示例代码 4.dup3函数原型 4.1 dup3函数 4.2…

rv1126人脸识别的相关操作

目录 一、代码的改写Makeflierkmedia_rockx_face_insert.cpprkmedia_rockx_face_rga_rtsp_main.cpprkmedia_rockx_face_two_rkisp_rtsp_main.cppsqlite3_operation.cpp二、在ubuntu上交叉编译三、板子上的相关操作一、代码的改写 Makeflie 修改交叉编译工具链 rkmedia_rockx_fa…

2023年NOC决赛-加码未来编程赛项决赛模拟题-Python模拟题--卷5

第一题 题目:输入一个整数n,计算其各位上数字之和,并用汉语写出每一位数字并输出。 【输入格式】一个整数 【输出格式】再一行内输出数字之和的每一位对应的汉字 【输入样例】1234 【输出样例】一零 第二题 题目:小溪使用 Excel 将任意 2 组数字中相同的数按照从小到…

终于有人把软件测试用例讲清楚了(一定要收藏)

目录 1&#xff1a;公司流程 1.1. 测试用例的4个特性 1.1. 测试用例通常包括以下几个组成元素&#xff1a; 1. 编写测试用例的基本方法 1.1.1. 概念 1.1.1. 示例 1.1练习案例: 1.1. 边界值法 1.1.1. 确定边界值的方法&#xff08;&#xff09; 1.1. 因果图法 1.1.1.…