Linux 线程同步和互斥-无名信号量

目录

信号量

信号量概念

pthread 库常用的信号量操作函数

pthread 库常用的信号量操作函数

 信号量 P 操作(减 1)

信号量 V 操作(加 1)

获取信号量的值

销毁信号量

无名信号量应用实例


信号量

信号量概念

线程间的信号量一般叫匿名信号量(进程间的信号量叫有名信号量),信号量广泛用于进程或线程间的同步和互斥,信号量本质上是一个非负的整数计数器,它被用来控制对公共资源的访问。
编程时可根据操作信号量值的结果判断是否对公共资源具有访问的权限:
当信号量值大于 0 时,则可以访问,否则将阻塞。
PV 原语是对信号量的操作,一次 P 操作使信号量减1,一次 V 操作使信号量加1。
信号量主要用于进程或线程间的同步和互斥这两种典型情况。
信号量用于互斥:

 

信号量用于同步:

 

POSIX 标准中,信号量分两种,一种是无名信号量,一种是有名信号量。无名信号量一般用于线程间同步或互斥,而有名信号量一般用于进程间同步或互斥。它们的区别和无名管道及有名管道的区别类似,无名信号量则直接保存在内存中,而有名信号量要求创建一个文件。

pthread 库常用的信号量操作函数

pthread 库常用的信号量操作函数

功能: 创建一个信号量并初始化它的值。一个无名信号量在被使用前必须先初始化。

 信号量 P 操作(减 1

阻塞函数

功能:将信号量的值减 1。
操作前,先检查信号量(sem)的值是否为 0,若信号量为 0,此函数会阻塞,直到信号量大于 0 时才进行减 1 操作。

        int sem_trywait(sem_t *sem); //非阻塞版本

以非阻塞的方式来对信号量进行减 1 操作。若操作前,信号量的值等于 0,则对信号量的操作失败,函数立即返回。

信号量 V 操作(加 1

 功能:将信号量的值加 1 并发出信号唤醒等待线程(sem_wait())。

获取信号量的值

int sem_getvalue(sem_t *sem, int *sval);
功能:
获取当前 sem 标识的信号量的值,保存在 sval 中。
0 表示资源可用 ,0 表示资源不可。

销毁信号量

int sem_destroy(sem_t *sem);
功能:
删除 sem 标识的信号量。

无名信号量应用实例

线程 1: 键盘输入字符串 遇到 quit 结束;
线程 2: 读取字符串打印输出;
#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <semaphore.h>
#include <strings.h>
#include <string.h>

sem_t  sem_g;	//输入进程信号量
sem_t  sem_p;	//打印进程信号量

char input[1024] = {'\0'};
int  ch = '\0';
int i = 0;

//输入进程
void * getchar_thread(void *arg)
{   //线程体:并发执行代码的入口这里开始

	while (1) {
		sem_wait(&sem_p);
		i = 0;
		printf("请录入字符串,结束请输入quit:");
		while((ch=getchar()) != EOF){
			if(ch == '\n'){
				continue;
			}
			input[i] = ch;
			i++;
			
			if(strstr(input,"quit") != NULL){
				input[i-4] = '\0';
				input[i-3] = '\0';
				input[i-2] = '\0';
				input[i-1] = '\0';
				break;
			}
				
		}

		sem_post(&sem_g);

	}
	return NULL;
}

//打印进程
void * print_thread(void *arg)
{   //线程体:并发执行代码的入口这里开始
	while(1) {
		sem_wait(&sem_g);

			printf("线程2打印:%s\n",input);	
			bzero(input, 1024);

		sem_post(&sem_p);

	}
	return NULL;
}

int main(void)
{
	
	pthread_t tid1, tid2;

	//初始化信号量  
	sem_init(&sem_p, 0,  1); //先输入
	sem_init(&sem_g, 0,  0); //后打印

	//创建2个线程
	pthread_create(&tid1, NULL, getchar_thread, NULL);
	pthread_create(&tid2, NULL, print_thread, NULL);
	
	//等待线程结束,回收线程资源
	pthread_join(tid1, NULL);
	pthread_join(tid2, NULL);
	

	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

宁静的海2006

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值