Linux——信号量、环形队列

news2024/12/27 12:59:49

Linux——信号量和环形队列

位图 (10)

文章目录

  • Linux——信号量和环形队列
      • 概念
      • 信号量的PV原语
      • 线程申请信号量失败将会被挂起
    • 信号量函数
      • sem_init初始化信号量
      • sem_destroy销毁信号量
      • sem_wait等待信号量
      • sem_post发布信号量
    • 基于环形队列的生产者消费者模型
      • 代码实现

概念

  • 临界资源:多线程执行流共享的资源就叫做临界资源

我们知道线程在操作临界资源时必须要进入临界区前先加锁,保证线程串行访问临界资源,避免出现多个线程同时访问临界资源而造成线程安全问题。而对临界区加锁,本质上是一个线程占用了整个临界资源,而实际上我们可以让整个临界资源分成不同的区域,让多个线程并发地访问不同区域,这样就能让多个线程同时访问临界资源而不会发生线程安全问题。这时候就需要引入信号量的概念

  • 信号量本质是一个计数器,属于无符号整数,可以用来衡量临界资源临界资源的多少
  • 执行流进入临界区前先申请信号量,对临界资源操作后,释放信号量。当线程申请到信号量时,意味着该线程在未来一定能够拥有临界资源的一部分,即申请信号量本质是对临界资源的某个区域进行了预定

image-20230724113614947

信号量的PV原语

  • 线程申请到信号量导致计数器sem–,该行为称为P原语。线程释放信号量导致计数器sem++,该行为称为V原语。
  • ​ 而每个线程都能够申请到信号量,意味着信号量本身作为公共资源,为了避免线程安全问题,信号量的PV原语必然具有原子性,也就是说信号量本身也作为临界资源。
  • P操作:将申请信号量的行为称为P操作,申请信号量的本质是申请临界资源中某个区域的使用权限,当申请成功时,该临界资源中的资源数目就会减一,本质上是计数器sem减一
  • V操作:将释放信号量的行为称为V操作,释放信号量的本质是归还临界资源某个区域的使用权限,当释放成功时,该临界资源的资源数目就会加一,本质上是计数器sem加一

线程申请信号量失败将会被挂起

当线程在申请信号量,若此时信号量sem为0,意味着申请的临界资源部分已经被全部被申请了,那么此时线程就在该信号量的队列中阻塞等待,直到信号量sem大于0时该线程才被释放。

  • 意味着信号量的本质是计数器,但信号量还包括一个资源等待队列

信号量函数

sem_init初始化信号量

需要注意的是:

  • 使用信号量需要链接pthread原生库
  • 信号量函数调用成功返回0,失败返回-1,错误信息存储在错误码
#include <semaphore.h>
int sem_init(sem_t *sem, int pshared, unsigned int value);
Link with -pthread.
  • 参数sem是信号量,需要传信号量的地址

  • pshared为0表示线程间共享,非零表示进程间共享

  • value为信号量初始值,即计数器sem的初始值

sem_destroy销毁信号量

#include <semaphore.h>
int sem_destroy(sem_t *sem);
Link with -pthread.
  • 参数sem是信号量,需要传信号量的地址

sem_wait等待信号量

 #include <semaphore.h>
int sem_wait(sem_t *sem);
  • 参数sem是信号量,需要传信号量的地址
  • 作用:等待信号量,若传入的信号量不为0,那么等待成功并将信号量sem减一,并且继续往下执行。若传入的信号量为0,那么调用函数的线程就阻塞在信号量等待队列,直到有线程释放了该信号量

sem_post发布信号量

#include <semaphore.h>
int sem_post(sem_t *sem);
  • 参数sem是信号量,需要传信号量的地址

  • 作用:发布信号量,表示资源使用完毕,可以归还资源了。将信号量值加一

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

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

image-20230724144943470

  • 对于生产者而言,看到的环形队列是空间资源,队列中有空余的空间才能往里push数据。那么可以将队列中的剩余空间定义成一个计数器即信号量_ spacesem,生产者通过获取_spacesem来对获取对队列操作权限

  • 对于消费者而言,看到的环形队列是数据资源,队列中有数据才能从中取到数据。那么可以将队列中的数据定义成一个计数器即信号量_ datasem,消费者通过获取_datasem来获取对队列的操作权限

  • 一开始队列里全是空余的位置,因此_ spacesem初始值为队列的空间数目,_datasem初始值为0

  • 大部分时候生产者和消费者都是并发执行的,除了以下两种情况:

  • 刚开始队列为空时,有可能在同一个位置,生产者往队列push数据,而消费者从队列中pop数据。但是生产者的信号量初始值为队列的容量,则P操作生效;而消费者的信号量初始值为0,则P操作失败。因此一开始只有生产者往队列中push数据。即生产者和消费者具有互斥关系
  • 当队列为满时,此时生产者和消费者都指向同一个位置。但此时队列的空余位置为0,那么生产者对应的信号量就为0从而P操作失败,进而阻塞等待;而队列中存在数据,消费者对应的信号量不为0那么消费者对应的P操作成功。因此队列为满时只有消费者从队列中pop数据。即此时生产者和消费者具有互斥关系

此外环形队列还有以下原则:

    1. 生产者和消费者不能同时对同一个位置进行访问,否则会造成数据不一致问题
    1. 消费者消费的位置不能超过生产者

image-20230724150848589

    1. 生产者生产的位置不能超过消费者

image-20230724151435474

代码实现

为了方便理解,以下以单生产者单消费者模型为例,生产者不停生产数据并往环形队列中存放,消费者不断的从队列中取数据进行消费

main.cc

void* productor(void* args)
{
   annulusqueue<int>* aq=static_cast<annulusqueue<int>*>(args);
    while(true)
    {
 int num=rand()%100;
aq->push(num);
cout<<"productor produce num: "<<num<<endl;
sleep(1);
    }
return nullptr;
}

void* consumer(void* args)
{
    annulusqueue<int>* caq=static_cast<annulusqueue<int>*>(args);
    while(true)
    {
        int ret;
        caq->pop(&ret);
        cout<<"consumer get num: "<<ret<<endl;
       // sleep(1);
    }
    return nullptr;
}

int main()
{
srand((unsigned int)time(nullptr)^getpid());//随机数种子
annulusqueue<int>* aq=new annulusqueue<int>();
    pthread_t p,c;//线程
    pthread_create(&p,nullptr,productor,aq);
    pthread_create(&c,nullptr,consumer,aq);
    

pthread_join(p,nullptr);
pthread_join(c,nullptr);

delete aq;
    return 0;
}

annulusqueue.hpp


#pragma once
#include<iostream>
#include<semaphore.h>
#include<vector>
#include<assert.h>
using namespace std;

static const int gmaxcp=5;
template<class T>
class annulusqueue
{
public:
annulusqueue(const int maxcp=gmaxcp):_maxcp(maxcp),_queue(maxcp)
{
    int n=sem_init(&_spacesem,0,_maxcp);//生产者以空间为信号量,那么初始空间即为队列容量
    assert(n==0);
    (void)n;
    int m=sem_init(&_datasem,0,0);//消费者以数据为信号量,那么初始的数据为0
    assert(m==0);
    (void)m;
_psetp=_cstep=0;//初始时,生产者和消费者都指向队列的开头即下标为0的位置

}
void P(sem_t &sem)
{
    int n=sem_wait(&sem);//若sem大于0则sem--并且往下走,若不满足条件则阻塞等待
    assert(n==0);//
    (void)n;
}
void V(sem_t &sem)
{
    int n=sem_post(&sem);//资源使用完成归还资源,sem++
    assert(n==0);
    (void)n;
}
void push(const T&in)
{
P(_spacesem);//对空间资源进行P操作即_spacesem--
_queue[_psetp++]=in;
_psetp%=_maxcp;
V(_datasem);//对数据资源进行V操作即_datasem++
}

void pop(T* out)
{
    P(_datasem);
    *out= _queue[_cstep++];
    _cstep%=_maxcp;
    V(_spacesem);
}

~annulusqueue()
{
    sem_destroy(&_spacesem);//销毁信号量
    sem_destroy(&_datasem);//销毁信号量
}

private:
sem_t _spacesem;//生产者对应的信号量--空间资源
sem_t _datasem;//消费者对应的信号量--数据资源
int _maxcp;//环形队列的容量
vector<T> _queue;//环形队列-实际上是数组
int _psetp;//生产者下标
int _cstep;//消费者下标
};
  • 当不设置环形队列的大小时,我们默认将环形队列的容量上限设置为5

  • 代码中的annulusqueue是用vector实现的,生产者每次生产的数据放到vector下标为 _ psetp的位置,消费者每次消费的数据来源于vector下标为_cstep的位置

  • 生产者每次生产数据后_ psetp都会进行++,标记下一次生产数据的存放位置,++后的下标会与环形队列的容量进行取模运算,实现“环形”的效果。

  • 消费者每次消费数据后_cstep都会进行++,标记下一次消费数据的来源位置,++后的下标会与环形队列的容量进行取模运算,实现“环形”的效果。

  • 生产者生产随机数,往队列中放数据,并且打印日志。消费者从队列中拿数据,并且打印日志。

  • 对于生产者而言,先要对空间资源进行P操作,即对队列中能使用的空间数目进行减一,_ spacesem–;对数据资源进行V操作,即队列中的数据量_datasem++加一

void push(const T&in)
{
P(_spacesem);//对空间资源进行P操作即_spacesem--
_queue[_psetp++]=in;
_psetp%=_maxcp;
V(_datasem);//对数据资源进行V操作即_datasem++
}
  • 对于消费者而言,先对数据资源进行P操作,即当前位置的数据被消费了,意味着队列中的数据消失了一份,_datasem --;对空间资源进行V操作,当前位置的数据被消费,意味着当前位置空余下来供生产者存放数据,即队列中的空间资源加一, _spacesem++
void pop(T* out)
{
    P(_datasem);//对数据资源进行P操作即_datasem--
    *out= _queue[_cstep++];
    _cstep%=_maxcp;
    V(_spacesem);//对空间资源进行V操作即_spacesem++
}
  • 生产者生产的慢,消费者消费的快。生产者每隔一秒生产一次,消费者不断的消费。结果是生产者生产一个数据消费者就消费一个数据

image-20230724160439670

  • 生产者生产的快,消费者消费的慢。生产者不停的生产,消费者每隔一秒消费一次。结果是生产者生产了队列空余位置数目的数据,然后消费者消费一个,生产者生产一个。并且消费者生产是按照生产时间从先往后消费
    image-20230724160803776

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

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

相关文章

Qt - 信号和槽

文章目录 信号和槽自定义信号和槽代码实现teacher 类申明信号方法student 添加槽并处理绑定信号和槽 当自定义信号和槽出现重载设置按钮点击 信号可以连接信号断开信号 disconnectQt4版本写法Lambda 表达式函数对象参数操作符重载函数参数可修改标示符函数返回值是函数体 总结拓…

【C++修炼之路】继承

&#x1f451;作者主页&#xff1a;安 度 因 &#x1f3e0;学习社区&#xff1a;StackFrame &#x1f4d6;专栏链接&#xff1a;C修炼之路 文章目录 一、概念及定义二、基类和派生类对象赋值转换三、继承中的作用域四、派生类的默认成员函数五、继承与友元六、继承与静态成员七…

2023最新谷粒商城笔记之Sentinel概述篇(全文总共13万字,超详细)

Sentinel概述 服务流控、熔断和降级 什么是熔断 当扇出链路的某个微服务不可用或者响应时间太长时&#xff0c;会进行服务的降级&#xff0c;**进而熔断该节点微服务的调用&#xff0c;快速返回错误的响应信息。**检测到该节点微服务调用响应正常后恢复调用链路。A服务调用B服…

服务器被挂病毒记录(redis入侵)

前言 今天正在快乐的打着游戏&#xff0c;突然一个浙江的电话&#xff1a; 好家伙&#xff0c;我那可怜的1核2g的服务器说在跑挖矿程序&#xff0c;苍天啊&#xff0c;大地呀&#xff0c;我那1核2g的服务器有啥跑呢&#xff0c;别难为这小家伙了。 解决过程&#xff1a; 1…

【学会动态规划】地下城游戏(10)

目录 动态规划怎么学&#xff1f; 1. 题目解析 2. 算法原理 1. 状态表示 2. 状态转移方程 3. 初始化 4. 填表顺序 5. 返回值 3. 代码编写 写在最后&#xff1a; 动态规划怎么学&#xff1f; 学习一个算法没有捷径&#xff0c;更何况是学习动态规划&#xff0c; 跟我…

计算机网络复习(路由器、交换机、广域网配置~)

文章目录 子网划分路由器和交换机的配置&#xff08;基础知识&#xff09;IOS基础IOS使用技巧Cisco设备的启动Cisco设备的配置途径配置文件的备份与恢复管理网络环境配置Telnet网络测试 配置路由表路由简介路由表简介路由的分类配置静态路由动态路由协议VLAN间的路由路由信息协…

Redis追本溯源(二)数据结构:String、List、Hash、Set、Zset底层数据结构原理

文章目录 一、String底层——sds&#xff08;Simple Dynamic String&#xff09;1.sds相比C语言字符串的优点2.结构3.扩容4.缩容 二、List底层——quickList、zipList1.quickList及其优化过程&#xff08;1&#xff09;quickList大致结构&#xff08;2&#xff09;引入zipList进…

Termux——安装配置

Termux简介1.基础知识1) 基础操作2) 目录结构3) 快捷键 2.基础命令1) 软件安装2) termux保持后台运行3&#xff09;[Process completed (signal 9) - press Enter] 问题修复4&#xff09;更换国内源5&#xff09;获取手机存储权限6&#xff09;基础软件安装 2. 美化相关1) 修改…

ChatGPT vs. 之前版本:性能与表现的对比

&#x1f337;&#x1f341; 博主 libin9iOak带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——libin9iOak的博客&#x1f390; &#x1f433; 《面试题大全》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33…

vue之ReadIdcardMD(身份证读取组件-移动设备)

组件功能 移动便携读取二代身份证信息组件(一般是非接刷二代证方式),包含无效身份证验证,过期身份证验证,是否满16周岁验证 (与windows二代证读取组件的区别是不会生成二代证图片,直接返回base64数据) #界面 #<

测等保2.0——安全区域边界

一、前言 今天我们来说说安全区域边界&#xff0c;顾名思义&#xff0c;安全区域边界就是保障网络边界处&#xff0c;包括网络对外界的边界和内部划分不同区域的交界处&#xff0c;我们的重点就是查看这些边界处是否部署必要的安全设备&#xff0c;包括防火墙、网闸、网关等安…

不止工具:音视频开发「利器」的新机遇

Boxing的制胜关键是快、准、稳&#xff0c;与“音视频开发”有异曲同工之妙。 数字化浪潮席卷、视频化形态加速、终端性能挑战加剧、端侧算力遭遇瓶颈...... 是否存在一种可能性&#xff0c;让所有企业从复杂的音视频开发工程中抽身&#xff0c;重新回归业务本身&#xff1f; …

vue项目启动npm run serve常见报错及解决办法

报错1&#xff1a; 如图&#xff1a; 解决方法&#xff1a;重新安装core-js , npm i core-js 报错2&#xff1a; Syntax Error: EslintPluginImportResolveError: unable to load resolver “alias”. 解决方法&#xff1a;npm install eslint-import-resolver-alias -D 报…

Raft 思想在架构中实践

Raft 诞生背景&#xff1a; 分布式存储系统通常通过维护多个副本来进行容错&#xff0c;提高系统的可用性。要实现此目标&#xff0c;就必须要解决分布式存储系统的最核心问题&#xff1a;维护多个副本的一致性。 首先需要解释一下什么是一致性&#xff08;consensus&#xf…

ABAP 为N的一个数,在原来基础上浮动在-30~30

需求&#xff1a;为N的一个数&#xff0c;在原来基础上浮动在-30~30 *&---------------------------------------------------------------------* *& Report ZZZZ111 *&---------------------------------------------------------------------* *& 需求&…

揉捻Map-疯狂Java

理论概述 定义 图&#xff08;Graph&#xff09;是由节点&#xff08;Vertex&#xff09;和连接节点的边&#xff08;Edge&#xff09;组成的一种非线性数 据结构。它用于描述事物之间的关系、连接或依赖。图是一种非线性的数据结构&#xff0c; 它广泛应用于计算机科学、数学…

Day_71-76 BP 神经网络

目录 一. 基础概念理解 1. 一点个人理解 2. 神经网络 二. bp神经网络的局部概念 1. 神经元 2. 激活函数 三. bp神经网络的过程 1. 算法流程图 2. 神经网络基础架构 2.1 正向传播过程 2.2 反向传播过程&#xff08;算法核心&#xff09; 四. 基本bp神经网络的代码实现 1. 抽象…

一文了解 MySQL 全新版本模型

MySQL 8.1 已经发布了&#xff0c;也宣布 MySQL 开始使用新的版本模型。 作者&#xff1a;Kenny Gryp / Airton Lastori MySQL 产品团队。 原文&#xff1a;https://blogs.oracle.com/mysql/post/introducing-mysql-innovation-and-longterm-support-lts-versions 引子 在 Ora…

边缘提取总结

边缘提取&#xff1a;什么是边缘&#xff1f; 图象的边缘是指图象局部区域亮度变化显著的部分&#xff0c;该区域的灰度剖面一般可以 看作是一个阶跃&#xff0c;既从一个灰度值在很小的缓冲区域内急剧变化到另一个灰度相 差较大的灰度值。 边缘有正负之分&#xff0c;就像…

React AntDesign表批量操作时的selectedRowKeys回显选中

不知道大家是不是在AntDesign的某一个列表想要做一个批量导出或者操作的时候&#xff0c;发现只要选择下一页&#xff0c;即使选中的ids 都有记录下面&#xff0c;但是就是不回显 后来问了chatGPT&#xff0c;对方的回答是&#xff1a; 在Ant Design的DataTable组件中&#xf…