当前位置: 首页 > news >正文

Linux:线程同步之信号量

信号量

(1)What(什么是信号量)

提供一种计数器的方式控制对共享资源的访问;当计数器大于0时,请求资源成功并计数器-1;当计数器小于0时,线程阻塞,等待其它线程执行signal(V操作)唤醒它

(2)Why(信号量的作用)

  • 实现线程的同步与互斥:通过信号量的设计,可以实现对共享资源的串行访问
  • 实现线程的等待与通知机制:当信号量小于0时,当前线程将被阻塞;当信号量大于0时,会唤醒一个阻塞在信号量上的线程

(3)How(如何使用信号量实现线程同步)

step01:创建信号量

可以看作是创建一个信号量对象

sem_t sem; 

step02:初始化信号量

可以看作是对信号量对象的一个初始化,这一过程会给信号量的计数器赋予一个初始值

int sem_init(sem_t *sem, int pshared, unsigned int value); 
  • 参数sem:被初始化的信号量对象
  • 参数pshared:默认为0,表示信号量用于线程同步;其它表示信号量用于进程同步
  • 参数value:表示信号量的数量,常用用于表示共享资源的数量
  • 返回值:成功返回0,失败返回-1,并设置错误码

step03:请求资源

请求获取共享资源,此时信号量的计数器减1;如果信号量小于1,请求失败,线程阻塞,直到信号量满足条件时解除阻塞

int sem_wait(sem_t *sem); 
  • 参数sem:请求共享资源,如果sem中的计数器大于0,则请求成功,否则线程阻塞
  • 返回值:成功返回0,失败返回-1,并设置错误码

step04:释放资源

释放共享资源,此时信号量的计数器加1,此时会唤醒一个等待该共享资源的线程

int sem_post(sem_t *sem);

  • 参数sem:释放共享资源,sem中的计数器+1
  • 返回值:成功返回0,失败返回-1,并设置错误码
    step05:销毁信号量

本质就是释放信号量对象的内存空间

int sem_destroy(sem_t *sem); 
  • 参数sem:将要被销毁的信号量对象
  • 返回值:成功返回0,失败返回-1,并设置错误码

(4)代码实例

以下代码是对共享资源的互斥访问,共享资源的个数为5

#include <semaphore.h>
#include <unistd.h>
#include <iostream>
#include <pthread.h>
using namespace std;
pthread_mutex_t mutex;
sem_t semProc;
sem_t semComu;
struct Node{int iVal;Node * ptrNext;static int iSize;
};
int Node::iSize = 0;
Node * head = NULL;
void *produce(void  *arg){int i = 0;while(1){sem_wait(&semProc);pthread_mutex_lock(&mutex);int iVal = i;Node *node = new Node;node->iVal = iVal;node->ptrNext = head->ptrNext;head->ptrNext = node;Node::iSize++;cout<<"Producing "<<iVal<<"("<<Node::iSize<<")"<<endl;pthread_mutex_unlock(&mutex);sem_post(&semComu);++i;sleep(1);}
}
void *consume(void *arg){while(1){sem_wait(&semComu);pthread_mutex_lock(&mutex);if(head->ptrNext!=NULL){cout<<"Comsuing "<<head->ptrNext->iVal;Node *ptr = head->ptrNext;head->ptrNext = ptr->ptrNext;Node::iSize--;delete ptr;cout<<"("<<Node::iSize<<")"<<endl;ptr = 0;}pthread_mutex_unlock(&mutex);sem_post(&semProc);sleep(2);return NULL; 
}
int main()
{head = new Node; pthread_t tidProc,  tidComu; sem_init(&semProc, 0, 5); sem_init(&semComu, 0, 0); pthread_create(&tidProc,  NULL,  produce,  NULL); pthread_create(&tidComu, NULL, conmuse, NULL); pthread_join(tidProc, NULL); pthread_join(tidComu, NULL); return 0; 
}
http://www.lryc.cn/news/418055.html

相关文章:

  • GPT-SoVITS-文本转语音(你的声音不再是唯一)
  • C语言深度剖析(部分)--剩下随缘更新
  • 计算机毕业设计选题推荐-电缆行业生产管理系统-Java/Python项目实战
  • Linux 下查看 CPU 使用率
  • 数理基础知识
  • Java解决lombok和mapstruct编译模块的问题
  • 大模型场景应用全集:持续更新中
  • 理解RabbitMQ中的消息存储机制:非持久化、持久化与惰性队列(Lazy Queue)
  • 【机器学习】BP神经网络正向计算
  • 谷粒商城实战笔记-108~109-elasticsearch删除与批量导入
  • RabbitMQ:发送者的可靠性之使用消息确认回调
  • HCIP学习 | OSPF---LSA限制、不规则区域、附录E、选路
  • CVE-2017-15715~Apache解析漏洞【春秋云境靶场渗透】
  • thinkphp 5.0.24生成模块
  • 值得注意!家里有带毛发动物就有浮毛?宠物空气净化器一键净化
  • Linux 代理(proxy)设置
  • 操作系统真相还原:获取文件属性
  • 聚鼎装饰画:投资一家装饰画店铺要花费多少钱
  • 编程的魅力、其重要性、学习方法以及未来趋势
  • ubuntu init set
  • MySQL数据分析进阶(八)存储过程
  • 最深的根,
  • 【常见的设计模式】工厂模式
  • postgres收缩工具两种工具的使用对比
  • 仿真入门——CST软件如何设置分布式计算的共享储存
  • 【JVM基础17】——实践-说一下JVM调优工具
  • 【QT】Qt中Websocket的使用
  • 【vue3】【elementPlus】【国际化】
  • 用python实现求两个整数的最大公约数
  • Linux 内核源码分析---proc 文件系统