10:信号量
10.1 信号量的存储结构
struct semid_ds
{
struct ipc_permsem_perm; 访问权限
struct sem *base; 信号量集合中信号量指针
unsigned short sem_nsems; 信号量集合中信号量成员的数目
time_t sem_otime; 最近调用semop的时间
time_t sem_ctime; 最近修改的时间
};
sem_base指向一个信号量数组,信号量由结构sem记载,
struct sem
{
unsigned short semval; 信号量取值
pid_t sempid; 最近访问进程ID
unsigned short semncnt; P阻塞进程数
unsigned short semzcnt Z阻塞进程数
}
10.2 信号量的创建
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semget(key_t key, int nsems, int semflg);
semget创建一个新的或者访问一个已经存在的信号量,key为关键字,nsems指定信号量
中信号量的个数,semflg的低9为确定了队列访问权限,
例:创建关键字为0x1234,访问权限0666的信号量集合,拥有10个信号量
int semid;
semid = semget(0x1234,10,0666|IPC_CREAT);
10.3 信号量控制
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semctl(int semid, int semnum,int cmd,union semun arg);
semctl对标识为semid的信号量集合中序号为semnum的信号量进行操作
union semun
{
int val; 信号量数值
struct semid_ds *buf; 信号量信息
ushort *array; 批量设置和赋值时使用
};
参数cmd的命令如下:
1.GETVAL: 获取semnum信号量的取值(semval)保存到arg.val
2.SETVAL: 设置semnum信号量的当前值为arg.val
3.GETPID: 获取semnum信号量的最近一次操作的进程的ID(sempid)到arg.val
4.GETNCNT: 获取semnum信号量的semncnt,保存到arg.val
5.GETZCNT: 获取semnum信号量的semzcnt,保存到arg.val
6.GETALL: 获取semid信号量集合中全部信号量的值,保存到arg.array
7.SETALL: 设置semid信号量集合中全部信号量的值为arg.val
8.IPC_STAT: 获取semid信号量集合的信息,保存到arg.buf
9.IPC_SET: 按照arg.buf设置semid的信号量集合的信息
10.IPC_RMID:删除semid的信号量集合
例1:获取标识号为semid的信号量集合的状态
union semnu semopts;
struct semid_ds semidbuf;
semopts.buf = &semidbuf;
semctl(semid,0,IPC_STAT,semopts);
例2:删除标识号为semid的信号量
semctl(semid,0,IPC_RMID);
例3:设置标识号为semid,序号为index的信号量取值为10
union semnu semopts;
semopts.val = 10;
semctl(semid,index,SETVAL,semopts);
10.4 信号量的操作
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semop(int semid,struct sembuf *sops,unsigned nsops);
semop对标识号为semid的信号量集合中的一个或多个信号量执行信号数值的
增加或比较操作,sops指向一个sembuf结构的缓冲区,其指定的具体的
操作信息,nsops指定了缓冲区中存储的sembuf结构元素的个数
struct sembuf
{
short sem_num; 信号量中的信号序号
short sem_op; 信号操作
short sem_flg; 操作方式:IPC_NOWAIT,SEM_UNDO
};
sem_num指定了信号量集合中待操作的信号序号
sem_op指定了操作的类型:
正整数 V操作,信号量数值增加sem_op
0 Z操作,判断信号量数值是否等于0
负整数 P操作,信号量值增加sem_op
sem_flg指定了操作的方式:
SEM_UNDO 进程结束时可自动回复原初始值
IPC_NOWAIT 进程非阻塞标志
例1:以阻塞方式对标识号为semid的信号量集合中序号为3的信号量执行p操作
struct sembuf sb;
sb.sem_num = 3;
sb.sem_op = -1
sb.sem_flg = sb.sem_flg & ~IPC_NOWAIT
semop(semid, &sb, 1);
例2:以阻塞方式对标识号为semid的信号量集合中序号为0的信号量执行v操作
struct sembuf sb;
sb.sem_num = 0;
sb.sem_op = 1;
sb.sem_flg = sb.sem_flg & ~IPC_NOWAIT
semop(semid,&sb,1);
10.1 信号量的存储结构
struct semid_ds
{
struct ipc_permsem_perm; 访问权限
struct sem *base; 信号量集合中信号量指针
unsigned short sem_nsems; 信号量集合中信号量成员的数目
time_t sem_otime; 最近调用semop的时间
time_t sem_ctime; 最近修改的时间
};
sem_base指向一个信号量数组,信号量由结构sem记载,
struct sem
{
unsigned short semval; 信号量取值
pid_t sempid; 最近访问进程ID
unsigned short semncnt; P阻塞进程数
unsigned short semzcnt Z阻塞进程数
}
10.2 信号量的创建
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semget(key_t key, int nsems, int semflg);
semget创建一个新的或者访问一个已经存在的信号量,key为关键字,nsems指定信号量
中信号量的个数,semflg的低9为确定了队列访问权限,
例:创建关键字为0x1234,访问权限0666的信号量集合,拥有10个信号量
int semid;
semid = semget(0x1234,10,0666|IPC_CREAT);
10.3 信号量控制
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semctl(int semid, int semnum,int cmd,union semun arg);
semctl对标识为semid的信号量集合中序号为semnum的信号量进行操作
union semun
{
int val; 信号量数值
struct semid_ds *buf; 信号量信息
ushort *array; 批量设置和赋值时使用
};
参数cmd的命令如下:
1.GETVAL: 获取semnum信号量的取值(semval)保存到arg.val
2.SETVAL: 设置semnum信号量的当前值为arg.val
3.GETPID: 获取semnum信号量的最近一次操作的进程的ID(sempid)到arg.val
4.GETNCNT: 获取semnum信号量的semncnt,保存到arg.val
5.GETZCNT: 获取semnum信号量的semzcnt,保存到arg.val
6.GETALL: 获取semid信号量集合中全部信号量的值,保存到arg.array
7.SETALL: 设置semid信号量集合中全部信号量的值为arg.val
8.IPC_STAT: 获取semid信号量集合的信息,保存到arg.buf
9.IPC_SET: 按照arg.buf设置semid的信号量集合的信息
10.IPC_RMID:删除semid的信号量集合
例1:获取标识号为semid的信号量集合的状态
union semnu semopts;
struct semid_ds semidbuf;
semopts.buf = &semidbuf;
semctl(semid,0,IPC_STAT,semopts);
例2:删除标识号为semid的信号量
semctl(semid,0,IPC_RMID);
例3:设置标识号为semid,序号为index的信号量取值为10
union semnu semopts;
semopts.val = 10;
semctl(semid,index,SETVAL,semopts);
10.4 信号量的操作
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semop(int semid,struct sembuf *sops,unsigned nsops);
semop对标识号为semid的信号量集合中的一个或多个信号量执行信号数值的
增加或比较操作,sops指向一个sembuf结构的缓冲区,其指定的具体的
操作信息,nsops指定了缓冲区中存储的sembuf结构元素的个数
struct sembuf
{
short sem_num; 信号量中的信号序号
short sem_op; 信号操作
short sem_flg; 操作方式:IPC_NOWAIT,SEM_UNDO
};
sem_num指定了信号量集合中待操作的信号序号
sem_op指定了操作的类型:
正整数 V操作,信号量数值增加sem_op
0 Z操作,判断信号量数值是否等于0
负整数 P操作,信号量值增加sem_op
sem_flg指定了操作的方式:
SEM_UNDO 进程结束时可自动回复原初始值
IPC_NOWAIT 进程非阻塞标志
例1:以阻塞方式对标识号为semid的信号量集合中序号为3的信号量执行p操作
struct sembuf sb;
sb.sem_num = 3;
sb.sem_op = -1
sb.sem_flg = sb.sem_flg & ~IPC_NOWAIT
semop(semid, &sb, 1);
例2:以阻塞方式对标识号为semid的信号量集合中序号为0的信号量执行v操作
struct sembuf sb;
sb.sem_num = 0;
sb.sem_op = 1;
sb.sem_flg = sb.sem_flg & ~IPC_NOWAIT
semop(semid,&sb,1);