【信号量】

本文介绍了信号量作为一种在多线程或多任务环境中控制共享资源访问的同步机制,包括信号量的工作原理、函数接口及其在示例中的应用。主要讨论了sem_init、sem_wait、sem_post等关键函数的作用和使用场景。
摘要由CSDN通过智能技术生成

什么是信号量?

信号量(Semaphore)是多线程或多任务环境中用于控制对共享资源的访问的一种机制。它允许多个线程或任务同时访问某些共享资源,但在进入共享资源的临界区之前,需要获取信号量。

当一个线程或任务尝试访问共享资源时,它首先会检查信号量的值。如果信号量的值为0,则该线程或任务会被阻塞,直到信号量的值变为大于0。一旦线程或任务获得了信号量(即信号量的值大于0),它就可以访问共享资源。在完成对共享资源的访问后,线程或任务会释放信号量,使其他等待的线程或任务可以继续执行。

因此,信号量是一种同步机制,用于控制多个线程或任务对共享资源的访问,避免发生竞态条件或死锁等问题。

信号量是相互排斥锁的升级版把相互排斥锁中1变成了n。举个简单的样例:如果如今有10个⼈,有⼀部⼿机。这10个⼈都竞争来使⽤⼿机打电话这就是相互排斥锁。对于信号量,如今可能是有4部⼿机,这10个⼈都竞争来使⽤⼿机打电话。相⽐相互排斥锁信号量由1变成了4。信号量相也就是操作系统中pv操作,它⼴泛应⽤进程或者线程间的同步与相互排斥。

信号量的接⼝函数

 信号量的函数都以sem_开头,线程中使⽤的基本信号量函数有4个,它们都声明在头⽂件semaphore.h中。

sem_init函数

该函数⽤于创建信号量,其原型如下:
int sem_init(sem_t *sem,int pshared,unsigned int value);

该函数初始化由sem指向的信号对象,设置它的共享选项,并给它⼀个初始的整数值。
pshared控制信号量的类型,如果其值为0,就表⽰这个信号量是当前进程的局部信号量,否则信号量就可以在多个进程之间共享,value为sem的初始值。调⽤成功时返回0,失败返回-1.

sem_wait函数

该函数⽤于以原⼦操作的⽅式将信号量的值减1。原⼦操作就是,如果两个线程企图同时给⼀个信号量加1或减1,它们之间不会互相⼲扰。
它的原型如下:
int sem_wait(sem_t *sem); 

sem_post函数

该函数⽤于以原⼦操作的⽅式将信号量的值加1。它的原型如下:
int sem_post(sem_t *sem);

释放信号量,让信号量的值加1。相当于V操作。与sem_wait⼀样,sem指向的对象是由sem_init调⽤初始化的信号量。调⽤成功时返回0,失败返回-1.

sem_destroy函数

 该函数⽤于对⽤完的信号量的清理。它的原型如下:
int sem_destroy(sem_t *sem);
成功时返回0,失败时返回-1.

sem_try_wait函数

相当于p操作。在信号量值⼤于0时都能将信号量的值减⼀,与上⾯sem_wait的差别是,在信号值⼩于0时。

int sem_try_wait(sem_t *sem)

sem_getvalue函数

⽤于得到信号量的值

int sem_getvalue(sem_t *sem)
 

信号量的函数使⽤

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <pthread.h>
#include <semaphore.h>
sem_t sem;
void func1(void* arg)
{
	sem_wait(&sem);
	int *running=arg;
	printf("thread running1\n");
	printf("%d\n",*running);
}
void func2(void* arg)
{
	printf("pthread2 running\n");
	sem_post(&sem);
}
int main()
{
	sem_init(&sem,0,0);
	pthread_t thread[2];
	int a=5;
	pthread_create(&(thread[0]),NULL,(void*)func1,(void*)&a);
	printf("main thread running\n");
	sleep(10);
	pthread_create(&(thread[1]),NULL,(void*)func2,(void*)&a);
	printf("main thread running2\n");
	pthread_join(thread[0],NULL);
	pthread_join(thread[1],NULL);
	sem_destroy(&sem);
	return 0;
}

⾸先创建信号量,第⼆个参数为0,表⽰这个信号量是当前进程的局部信号量,初始值为0。
然后使⽤pthread_create()函数创建两个线程,传⼊参数a。
线程1创建完成后,由于信号量初始化value=0,调⽤sem_wait会阻塞这个线程,信号量的值将减少1(此时<0),这个线程函数就会等待。
主线程sleep(10)后,线程2创建并执⾏调⽤,sem_post⽤于使信号量的值增加1,此时线程1可以执⾏。
使⽤pthread_join等待所创建的两个线程的结束。

编译:
gcc testSem.c -o testSem -lpthread
./testSem

输出:
main thread running
main thread running2
pthread2 running
thread running1

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值