linux信号量初识
以下程序使用信号量控制程序运行
“信号”量
“变”量
/*信号量(semaphore)是变量,是一种特殊的变量。它仅取正值。
对信息号量的操作只有2种:等待(wait)和发送信号(signal)*/
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <unistd.h>
#include <errno.h>
#define IPCKEY 0x111
#define LOCKFILE "/home/liuyi/comm/1.lock"
/*ftok的生成的key值依赖IPCKEY和LOCKFILE生成*/
/*arg for semctl systemcalls.*/
union semun {
int val;/*value for SETVAL*/
struct semid_ds *buf;/*buffer for IPC_STAT&IPC_SET*/
unsigned short *arrary;/*array for GETALL&SETALL*/
};
/*结构体说明
struct sembuf
{
unsigned short sem_num;
操作信号在信号集中的索引,第一个信号的索引是0
short sem_op;
如果其值为正数,
该值会加到现有的信号内含值中。通常用于释放所控资源的使用权;
如果sem_op的值为负数,会减去改信号内含的相应值,而其绝对值又大于信号的现值,
操作将会阻塞,直到信号值大于或等于sem_op的绝对值。通常用于获取资源的使用权;
如果sem_op的值为0,则操作将暂时阻塞,直到信号的值变为0。
short sem_flg;
信号操作标志,
IPC_NOWAIT 对信号的操作不能满足时,semop()不会阻塞,并立即返回,同时设定错误信息
SEM_UNDO 程序结束时(不论正常或不正常),保证信号值会被重设为semop()调用前的值。
这样做的目的在于避免程序在异常情况下结束时未将锁定的资源解锁,造成该资源永远锁定。
(重要的地方)
}; */
int main()
{
key_t key;
int sem_id;
union semun options;
struct sembuf lock_it;/*sembuf见说明*/
key = ftok(LOCKFILE,IPCKEY);
printf("key=[0x%x]\n",key);
sem_id = semget(key,1,IPC_CREAT|IPC_EXCL|0666);/*创建信号量集*/
/*key,刚才ftok产生的,若指定IPC_PRIVATE,则由系统分配。*/
/*1,信号量集中多少个信号量。*/
/*IPC_CREAT如果信号量集在系统内核中不存在,则创建信号量集。
IPC_EXCL当和 IPC_CREAT一同使用时,如果信号量集已经存在,则调用失败。*/
/*0666,该信号权限为666*/
/*ps:可以用ipcs –s命令来查看是否创建成功。ipcrm -s semid号来删除指定的信号量集。*/
printf("创建信号量集ID为[%d]\n",sem_id);
if(sem_id<0)
{
printf("信号量已经存在\n");
sem_id = semget(key,1,IPC_CREAT|0666);/*已经存在信号量集,取得信号量集ID*/
if(-1 == sem_id)
{
printf("取已存在的信号量集ID错误\n");
exit(-1);
}
printf("已存在的信号量集ID为[%d]\n",sem_id);
}
else
{
options.val =1;/*设信号量集中的 那个信号量值为1*/
semctl(sem_id,0,SETVAL,options);/*操作信号量集,SETVAL这里表示设置值*/
/*sem_id:信号量ID,也就是semget函数的返回值*/
/*0:要操作信号量在信号量集中的索引,从0开始*/
/*SETVAL:操作类型,SETALL设置信号量集中的所有的信号量的值,
SETVAL设置信号量集中的一个单独的信号量的值。等等*/
/*options:要设定信号量集中的信号量值 的联合体,可选*/
}
printf("信号集ID sem_id=[%d] \n",sem_id);
int i = semctl(sem_id,0,GETVAL);/*获取索引为0的位置的信号量的值*/
printf("1索引为0的位置的信号量的值为[%d]\n",i);
lock_it.sem_num =0; /*索引*/
lock_it.sem_op = -1;/*设成-1,就是将信号量的值减去1*/
lock_it.sem_flg = SEM_UNDO;/*程序结束后还原信号量的值*/
semop(sem_id,&lock_it,1);/*操作信号量,使信号量的值减去abs(lock_it.sem_op)*/
/*ps:如果当前信号量的值小于abs(lock_it.sem_op),会阻塞,直到当前信号量的值不
小于abs(lock_it.sem_op)时候才会执行(靠外部程序改变当前信号量的值)*/
i = semctl(sem_id,0,GETVAL);/*打印操作后的信号量值*/
printf("2索引为0的位置的信号量的值为[%d]\n",i);
/*这里开始就是你干的私活了*/
int j;
scanf("%d",&j);
printf("finish\n");
return 0;
}