信号量相当于记录资源能同时被多少个进程访问。
信号量的作用:进程间同步控制。
信号量有一个初值,每当有进程申请使用信号量,就会通过一个P操作对信号量进行-1操作,当计数器减到0的时候就说明没有资源了,其他进程要想访问就必须等待,当该进程执行完这段工作之后,就会执行V操作,即对信号量进行+1操作。
头文件:
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
信号量的创建
函数原型:int semget(key_t key, int nsems, int semflg);
其中,key_t一是个标识数据结构唯一的key键值,可以给让内核自动给,也可以自己调用ftok函数绑定一个;nsems是指创造出多少个信号量,信号量集。semflg创建消息队列的参数,有IPC_CREAT 和 IPC_EXCL。
信号量操作:P(-1)操作,V(+1)操作.
•函数原型:int semop(int semid, struct sembuf *buf, unsigned lenth);
第一个semid内核对象
第二个参数是数组名结构体
第三个参数表示同时操作几个信号量,数组大小。
信号量的删除
函数原型:int smectl(int semid,int pos,int cmd,/union semun un/)
semid:内核对象。
pos:信号量数组中哪一个信号,即数组下标。
cmd:所采取的操作。
union semun un:可有可无。
sem.h
#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <assert.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semid;
union semun
{
int val;
};
void sem_get(); //创建或者获取信号量
void sem_p(); //P操作
void sem_v(); //V操作
void sem_del(); //删除
#include "sem.h"
void sem_get()
{
semid = semget((key_t)1234, 1, 0664);
if(semid == -1) // 获取失败
{
semid = semget((key_t)1234, 1, 0664 | IPC_CREAT);
assert(semid != -1);
union semun v;
v.val = 0;
if(semctl(semid, 0, SETVAL, v) < 0)
{
perror("error");
exit(0);
}
}
}
void sem_p()
{
struct sembuf buff;
buff.sem_num = 0;
buff.sem_op = -1;
buff.sem_flg = SEM_UNDO;
if(-1 == semop(semid, &buff, 1))
{
perror("p error");
exit(0);
}
}
void sem_v()
{
struct sembuf buff;
buff.sem_num = 0;
buff.sem_op = 1;
buff.sem_flg = SEM_UNDO;
if(semop(semid, &buff, 1) < 0)
{
perror("p error");
exit(0);
}
}
void sem_del()
{
if(-1 == semctl(semid, 0, IPC_RMID))
{
perror("del error");
exit(0);
}
}