信号量通信

来源:互联网 发布:zebra标签打印软件 编辑:程序博客网 时间:2024/06/06 03:59

在介绍信号量通信之前,首先需要明确几个概念:

1、临界资源:

同一时刻只能被一个进程访问

2、临界区:

访问临界资源的代码区域

3、原子操作:

任何情况下都不能被打断的操作,只能等完成该操作再进行其他操作。

4、内核对象:

用于进程间通讯时,多进程能够访问同一资源的记录。


如果有多个进程要访问临界资源,但同一时刻只能有一个进程访问,这时总要有一个先后顺序。信号量的作用也就是进程间同步控制。信号量与锁不同,信号量也可以控制伪临界资源(信号量控制的资源可以被多个进程访问,只是有限制),例如停车场,一般停车场可以停放多辆车,只是有数量限制,会有提示在某一时刻还可以有几辆车进去停放。也就是说,信号量记录资源能同时被几个进程访问,它是一个计数器。信号量并非单个值,而是含有一个或多个信号量值的集合。


信号量操作:
1、创建或获取:如果是创建必须初始化,如果获取则不能初始化。 

int semget(key_t key,int nsems,int flag);

参数:key用户标识,nsems创建多少个,flag操作模式及操作权限(创建时)。

成功返回信号量ID,失败返回-1。
2、减一操作(p操作):
3、加一操作(v操作):

int semop(int semid,struct semnuf*buf,int lenth);

参数:在semid指定的信号量集操作,buf数组名(要操作哪些),lenth数组长度

成功返回0,失败返回-1。

4、删除:内核对象创建出来就存在,不使用时要删除

int semctl(int semid,int semnum,int cmd,/*union semun un*/);

参数:在semid指定的信号量集操作,semnum指定信号量集中的一个,cmd采取什么操作,第四个参数是可选的

该函数用于初始化、删除。

如果要创建还要初始化就需要2个函数一起,进行p操作或v操作之后获取也需要2个函数一起,考虑到类似情况比较麻烦,那么可以对其封装(《信号量有关函数的封装》):
sem_get();//获取信号量 如果第一次初始化
sem_p();//p操作
sem_v();//v操作
sem_del();//删除信号量