线程同步与互斥:POSIX无名信号量
信号量概述
信号量广泛用于进程或线程间的同步和互斥,信号量本质上是一个非负的整数计数器,它被用来控制对公共资源的访问。
编程时可根据操作信号量值的结果判断是否对公共资源具有访问的权限,当信号量值大于 0 时,则可以访问,否则将阻塞。PV 原语是对信号量的操作,一次 P 操作使信号量减1,一次 V 操作使信号量加1。
信号量主要用于进程或线程间的同步和互斥这两种典型情况。
信号量用于互斥:
信号量用于同步:
在 POSIX 标准中,信号量分两种,一种是无名信号量,一种是有名信号量。无名信号量一般用于线程间同步或互斥,而有名信号量一般用于进程间同步或互斥。它们的区别和管道及命名管道的区别类似,无名信号量则直接保存在内存中,而有名信号量要求创建一个文件。
无名信号量基本操作
#include <semaphore.h>
注意:编译信号量操作函数时,需要加上参数-lpthread。
信号量数据类型为:sem_t。
1)初始化信号量
int sem_init(sem_t *sem, int pshared, unsigned int value);
功能:
创建一个信号量并初始化它的值。一个无名信号量在被使用前必须先初始化。
参数:
sem:信号量的地址。
pshared:等于 0,信号量在线程间共享(常用);不等于0,信号量在进程间共享。
value:信号量的初始值。
返回值:
成功:0
失败:-1
2)信号量 P 操作(减 1)
int sem_wait(sem_t *sem);
功能:
将信号量的值减 1。操作前,先检查信号量(sem)的值是否为 0,若信号量为 0,此函数会阻塞,直到信号量大于 0 时才进行减 1 操作。
参数:
sem:信号量的地址。
返回值:
成功:0
失败:-1
int sem_trywait(sem_t *sem);
以非阻塞的方式来对信号量进行减 1 操作。若操作前,信号量的值等于 0,则对信号量的操作失败,函数立即返回。
3)信号量 V 操作(加 1)
int sem_post(sem_t *sem);
功能:
将信号量的值加 1 并发出信号唤醒等待线程(sem_wait())。
参数:
sem:信号量的地址。
返回值:
成功:0
失败:-1
4)获取信号量的值
int sem_getvalue(sem_t *sem, int *sval);
功能:
获取 sem 标识的信号量的值,保存在 sval 中。
参数:
sem:信号量地址。
sval:保存信号量值的地址。
返回值:
成功:0
失败:-1
5)销毁信号量
int sem_destroy(sem_t *sem);
功能:
删除 sem 标识的信号量。
参数:
sem:信号量地址。
返回值:
成功:0
失败:-1
无名信号量应用实例
- #include <stdio.h>
- #include <pthread.h>
- #include <unistd.h>
- #include <semaphore.h>
- sem_t sem; //信号量
- void printer(char *str)
- {
- sem_wait(&sem);//减一
- while(*str)
- {
- putchar(*str);
- fflush(stdout);
- str++;
- sleep(1);
- }
- printf("\n");
- sem_post(&sem);//加一
- }
- void *thread_fun1(void *arg)
- {
- char *str1 = "hello";
- printer(str1);
- }
- void *thread_fun2(void *arg)
- {
- char *str2 = "world";
- printer(str2);
- }
- int main(void)
- {
- pthread_t tid1, tid2;
- sem_init(&sem, 0, 1); //初始化信号量,初始值为 1
- //创建 2 个线程
- pthread_create(&tid1, NULL, thread_fun1, NULL);
- pthread_create(&tid2, NULL, thread_fun2, NULL);
- //等待线程结束,回收其资源
- pthread_join(tid1, NULL);
- pthread_join(tid2, NULL);
- sem_destroy(&sem); //销毁信号量
- return 0;
- }
运行结果如下:
信号量用于同步实例:
- #include <stdio.h>
- #include <unistd.h>
- #include <pthread.h>
- #include <semaphore.h>
- sem_t sem_g,sem_p; //定义两个信号量
- char ch = 'a';
- void *pthread_g(void *arg) //此线程改变字符ch的值
- {
- while(1)
- {
- sem_wait(&sem_g);
- ch++;
- sleep(1);
- sem_post(&sem_p);
- }
- }
- void *pthread_p(void *arg) //此线程打印ch的值
- {
- while(1)
- {
- sem_wait(&sem_p);
- printf("%c",ch);
- fflush(stdout);
- sem_post(&sem_g);
- }
- }
- int main(int argc, char *argv[])
- {
- pthread_t tid1,tid2;
- sem_init(&sem_g, 0, 0); //初始化信号量
- sem_init(&sem_p, 0, 1);
- pthread_create(&tid1, NULL, pthread_g, NULL);
- pthread_create(&tid2, NULL, pthread_p, NULL);
- pthread_join(tid1, NULL);
- pthread_join(tid2, NULL);
- return 0;
- }
运行结果如下:
本教程示例代码下载请点此处。
线程同步与互斥:POSIX无名信号量相关推荐
- java 信号量 互斥锁_线程同步(互斥锁与信号量的作用与区别)
"信号量用在多线程多任务同步的,一个线程完成了某一个动作就通过信号量告诉别的线程,别的线程再进行某些动作(大家都在semtake的时候,就阻塞在 哪里).而互斥锁是用在多线程多任务互斥的,一 ...
- c++ linux 线程等待与唤醒_C++ Linux线程同步机制:POSIX信号量,互斥锁,条件变量...
线程同步机制:POSIX 信号量,互斥量,条件变量 POSIX 信号量 常用的POSIX 信号量函数为如下5个: sem_init sem_destroy sem_wait sem_trywait s ...
- linux线程基础篇----线程同步与互斥
linux线程基础----线程同步与互斥 一.同步的概念 1.同步概念 所谓同步,即同时起步,协调一致.不同的对象,对"同步"的理解方式略有不同.如,设备同步,是指在两个设备 ...
- C++ 线程同步之互斥锁
文章目录 1.简介 2.std::mutex 3.线程同步 4.std::lock_guard 5.std::recursive_mutex-少用 6.std::timed_mutex 1.简介 进行 ...
- 【Linux系统编程】线程同步与互斥:POSIX无名信号量
信号量概述 信号量广泛用于进程或线程间的同步和互斥,信号量本质上是一个非负的整数计数器,它被用来控制对公共资源的访问. 编程时可根据操作信号量值的结果判断是否对公共资源具有访问的权限,当信号量值大于 ...
- c/c++:线程同步(互斥锁、死锁、读写锁、条件变量、生产者和消费者模型、信号量)
目录 1. 概念 2. 互斥锁 3. 死锁 4. 读写锁 5. 条件变量 5.1 生产者和消费者模型 6. 信号量 1. 概念 线程同步: > 当有一个线程在对内存进行操作时,其他线程都不可以对 ...
- 【Linux系统编程】POSIX无名信号量
00. 目录 文章目录 00. 目录 01. 信号量概述 02. 相关函数 2.1 初始化信号量 2.2 销毁信号量 2.3 信号量P操作(减1) 2.4 信号量V操作(加1) 2.5 获取信号量的值 ...
- Linux系统开发 | 线程同步与互斥
拥有梦想是一种智力,实现梦想是一种能力. 概述 线程共享同一进程的地址空间,线程间通过全局变量交换数据进行通信.为了避免多个进程同时访问同一个共享资源而引发冲突,多个线程访问共享数据时需要同步或互斥机 ...
- Linux 多线程同步机制:互斥量、信号量、条件变量
互斥量:互斥量提供对共享资源的保护访问,它的两种状态:lock和unlock,用来保证某段时间内只有一个线程使用共享资源,互斥量的数据类型是pthread_mutex_t 主要涉及函数:pthread ...
- 1线程同步:互斥量,死锁
1线程为什么要同步 A:共享资源,多个线程都可对共享资源操作. B:线程操作共享资源的先后顺序不确定. C:处理器对存储器的操作一般不是原子操作. 2互斥量 mutex操作原语 pthread_ ...
最新文章
- 模拟简单计算器及比较大小
- 计算机一级查找同类型文件,如何快捷找出电脑内的重复文件
- 微软AirSim,一个无人机和机器人的模拟器
- MCITP第十二课——642课件2新知识点笔记
- CSS 水平对齐 text-align-last属性
- 老板催你交圣诞海报设计?PSD源文件素材,直接编辑
- MVC之LayOut布局页
- matlab cy11,matlab解线性回归方程,y=a0+a1*A+a2*B+a3*C+a4*D+a5*E+a6*F; 数据足够,求a0,a
- 关于linux安装openoffice无法启动
- 二十五、JAVA多线程(三、线程同步)
- 到爱尔兰敲代码 / Come, Coding in Ireland
- 华为笔试题库及性格测试
- 硬件电路设计之VGA ADL5201
- java制作手机投射电脑_将Android手机投影到Win10计算机的最简单教程
- Git项目管理修改项目名称
- html ul怎么去掉内边距,ul默认有内边距
- Html设置表格撑开,【CSS】表格或div被撑开的解决办法
- acm测试题目数据生成
- allergro音乐术语什么意思_rit(这是音乐术语)什么意思?
- ibm r40的内存品牌_在IBM Business Process Manager中品牌化和定制教练主题
热门文章
- CodeForces - 982C Cut 'em all!
- [bzoj 1855][SCOI2010]股票交易
- C#6.0新特性的尝试
- ASP.NET 取得 Request URL 的各个部分
- web安全day39:渗透测试方法论
- 在微型计算机中1G标准等于,微型计算机的基础知识概要.ppt
- Ansible详解(十)——Ansible Template模板基础
- Tomcat详解(四)——Tomcat配置详解
- 习惯性的对自己没信心,其实没想到自己已经变强大了。
- 菠萝在线完成数千万元A轮融资,盛力投资领投